青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

Error

C++博客 首頁 新隨筆 聯(lián)系 聚合 管理
  217 Posts :: 61 Stories :: 32 Comments :: 0 Trackbacks
首先一個原則是這樣:被用來delete的指針,一定是new出來的。
在設(shè)計智能指針的時候發(fā)現(xiàn),如果采用delete this機制,到最后可能會有嚴重的問題。
假設(shè)有這樣的多重繼承:
class CA{};
class CB{};
class CC{};
class CABC : public CA, public CB, public CC
{};
CABC* pAbc = new CABC;
CA* pA = pAbc;
CB* pB = pAbc;
CC* pC = pAbc;
A B C指針的值都是不一樣的,如果是簡單的使用delete 來刪除,問題就大條了,pB pC的地址和new出來的pAbc不一樣,這樣就出現(xiàn)堆錯誤。
如果設(shè)計入侵式RefObj問題就大條了:
class Ref
{
public:
virtual void Test()
{
std::cout << this << std::endl;
}
int m_i;
};
class CA : public virtual Ref{};
class CB : public virtual Ref{};
class CC : public virtual Ref{};
class CABC : public CA, public CB, public CC{};
         C* pAbc = new CABC;
CA* pA = pAbc;
CB* pB = pAbc;
CC* pC = pAbc;
通過打印可以看到,雖然ABC 在基類Test打印出來的this地址一致,但是和實際正確的delete指針(new出來的)是不一致的,這就準備堆錯誤了,,,
這基本意味著使用入侵式樣Ref的類,使用了多重繼承就是作死。除非和com的思想一樣,Ref不是作為基類處理,而是作為純虛接口,但是又會使得編碼復雜化,基本上全部要用組合來替代繼承
class IXXX;
class CXXX;
class IYYY;
class CYYYXXX : public CXXX, public IYYY;
這樣的用法無法實現(xiàn),應(yīng)為沒有實現(xiàn)IXXX接口,是一個虛基類,實現(xiàn)了在調(diào)用的時候也是各種基類未指定的錯誤。

奇怪的是shared_ptr沒有這個限制,猜測是在第一次從裸指針構(gòu)造出來的時候,保留了原始地址,生成了一個析構(gòu)函數(shù),會在shared_ptr之間共享,直到最后一個對象析構(gòu)的時候調(diào)用.
實際觀察std::shared_ptr的實現(xiàn),發(fā)現(xiàn)其構(gòu)造函數(shù)不是控制一定要轉(zhuǎn)換成T指定的類型,而是給入?yún)?shù)的實際類型,后面的管理也分兩個部分,一層得到根據(jù)T轉(zhuǎn)換后的值,實際的生命周期管理得到仍然是實際類型,所以他delete的時候還是當初給定的值。
所以,感覺智能指針不是什么好東西,背后的細節(jié)太多,一個不小心就死翹翹



class RefObj
{
public:
RefObj() : m_i(0){}
public:
void Increate(){m_i++;}
void Decreate()
{
if(0 == --m_i)
{
delete this;
}
}
private: 
int m_i;
};
class BaseA : public virtual RefObj {public: int m_iA;};
class BaseB : public virtual RefObj {public: int m_iB;};
class BaseC : public virtual RefObj {public: int m_iC;};
class CBaseABC : public BaseA,
public BaseB,
public BaseC
{
};
void Test(BaseB* pB)
{
pB->Decreate();
}
int _tmain(int argc, _TCHAR* argv[])
{
ETcpSocket tcpSocket;
CBaseABC* pABC = new CBaseABC;
pABC->Increate();
BaseB* pB = pABC;
Test(pB);
return 0;
}



始終還是覺得相比shared_ptr,RefObj這種,還是有部分場景更加格式,至少不用擔心類型轉(zhuǎn)換變得異常麻煩,,,
后發(fā)現(xiàn)如下實現(xiàn)方式:
class IRefObj
{
public:
virtual void __DecRef() = 0;
};
template<typename TType>
class TRefObj : public IRefObj
{
public:
void __DecRef()
{
delete reinterpret_cast<TType*>(this);
}
};
class CRefTest : public TRefObj<CRefTest>
{
};
class CRefTestA : public CRefTest
{
};
class CTestA {int i;};
class CTestB {int j;};
class CTestC : public CTestB, public CRefTestA, public  CTestA
{
};
int _tmain(int argc, _TCHAR* argv[])
{
CTestC* pTC = new CTestC;
CTestB* pTB = pTC;
CRefTestA* pRTA = pTC;
CTestA* pTA = pTC;
pRTA->__DecRef();

這個時候CRefTest是預期額值,懷疑可能和編譯器有關(guān),VS測試OK,gcc沒去實驗。
分析了一下原因:C++保證單根繼承的時候基類和派生類地址是一樣的,如果是多重繼承,那么也保證和最深的根父類地址樣,順便的,從最深的根節(jié)點,選一路下來是安全的。
class CRefTest : public TRefObj<CRefTest>所以約定TRefObj<CRefTest>這玩意和永遠和接口在一個級別,可以保證IRefObj永遠是最深的根,就安全了?
目前只在控件設(shè)計的時候用RefObj吧,,,坑的比較深,,,
posted on 2014-10-16 16:29 Enic 閱讀(781) 評論(2)  編輯 收藏 引用 所屬分類: C/C++技巧

評論

# re: C++多重繼承導致delete引起堆錯誤、智能指針設(shè)計陷阱[未登錄] 2014-10-17 10:58 Chipset
這種對象指針默認轉(zhuǎn)換中類型都丟了,析構(gòu)的時候調(diào)用哪個系夠函數(shù)?手工刪除時就死悄悄,用智能指針當然也照樣死悄悄。何止對象這樣啊,C++里永遠都是這樣,如果自己想自殺,C++絕對成全你,這是Human Right!  回復  更多評論
  

# re: C++多重繼承導致delete引起堆錯誤、智能指針設(shè)計陷阱 2014-10-30 14:19 Enic
用boost::shared_ptr 或者std::shared_ptr的時候沒有這個問題,我猜測是共享數(shù)據(jù)卡里邊保留了原始指針,,,@Chipset
  回復  更多評論
  

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            久久偷看各类wc女厕嘘嘘偷窃| 国产一区二区0| 亚洲新中文字幕| 亚洲美女中文字幕| 亚洲人www| 亚洲免费成人| 亚洲色图综合久久| 亚洲主播在线观看| 欧美中在线观看| 看片网站欧美日韩| 亚洲日本理论电影| 亚洲美女av电影| 亚洲一区二区三区精品视频| 欧美一区二区三区在线| 久久亚洲国产精品一区二区 | 亚洲一区二区在| 午夜精品影院| 鲁鲁狠狠狠7777一区二区| 亚洲高清不卡一区| 99香蕉国产精品偷在线观看| 亚洲一区二区三区精品动漫| 久久九九免费视频| 欧美日韩一二三四五区| 国产精品一区=区| 亚洲国产精品第一区二区| 亚洲欧美不卡| 欧美激情第二页| 中文高清一区| 免费不卡亚洲欧美| 国产视频一区二区在线观看| 亚洲每日更新| 久久这里只有| 亚洲影视九九影院在线观看| 欧美成人久久| 在线播放中文字幕一区| 午夜日韩av| av成人手机在线| 欧美精品播放| 亚洲第一在线综合在线| 欧美在线日韩精品| 亚洲精品视频免费观看| 久久青青草原一区二区| 国产农村妇女毛片精品久久莱园子| 亚洲精品国产系列| 免费在线日韩av| 久久精品欧美日韩精品| 国产欧美精品日韩区二区麻豆天美| 日韩午夜电影| 亚洲国产成人精品女人久久久| 久久精品国产99国产精品澳门| 国产精品伦一区| 亚洲一区精彩视频| 一区二区高清在线| 欧美日韩天堂| 一区二区三区精品在线| 亚洲人成免费| 欧美了一区在线观看| 日韩视频免费在线| 亚洲精品在线视频| 欧美了一区在线观看| 欧美99久久| 国产精品热久久久久夜色精品三区| 亚洲高清视频一区| 久久只精品国产| 久久午夜国产精品| 亚洲国语精品自产拍在线观看| 久久精品国产一区二区电影| 一区二区精品| 国产精品成av人在线视午夜片| 国产精品99久久久久久久久久久久 | 欧美视频手机在线| 亚洲视频axxx| 亚洲一区二区三区精品视频| 国产欧美日韩在线视频| 久久精品亚洲精品国产欧美kt∨| 亚洲欧美日韩视频二区| 国产一区二区三区精品欧美日韩一区二区三区 | 新67194成人永久网站| 亚洲一级在线观看| 国产欧美激情| 蜜桃伊人久久| 欧美精品三区| 亚洲欧美一区二区三区在线 | 亚洲免费婷婷| 国内偷自视频区视频综合| 欧美高清视频免费观看| 欧美激情亚洲激情| 午夜精品影院在线观看| 久久久久国产精品一区二区| 亚洲精品社区| 亚洲欧美日韩综合国产aⅴ| 影音先锋一区| 亚洲最新色图| 红桃视频成人| 一本大道av伊人久久综合| 狠狠色综合色综合网络| 99国内精品| 影院欧美亚洲| 亚洲一区二区免费看| 亚洲国产日韩综合一区| 亚洲影院色无极综合| 亚洲激情六月丁香| 亚洲免费中文字幕| 99xxxx成人网| 久久蜜桃av一区精品变态类天堂| 在线亚洲+欧美+日本专区| 欧美一区二区三区男人的天堂| 91久久久久久久久| 欧美一区二区日韩一区二区| 一本色道久久综合亚洲精品婷婷| 性8sex亚洲区入口| 宅男精品视频| 免费在线视频一区| 久久午夜影视| 国产伦精品一区二区三区四区免费| 欧美激情一区在线| 妖精视频成人观看www| 激情综合色综合久久| 欧美一区网站| 麻豆精品在线播放| 国产精品久久午夜夜伦鲁鲁| 快播亚洲色图| 国产乱码精品一区二区三区忘忧草| 欧美aa国产视频| 国产欧美日韩亚洲精品| 欧美在线视频导航| 欧美日韩亚洲一区二区三区| 欧美刺激性大交免费视频| 欧美日韩亚洲网| 国产精品亚发布| 在线国产精品一区| 亚洲一区二区在线| 久久久久久国产精品mv| 久久久噜噜噜久久中文字幕色伊伊| 国产日韩欧美一二三区| 噜噜噜噜噜久久久久久91| 欧美jizzhd精品欧美喷水| 国产人妖伪娘一区91| 在线视频欧美精品| 欧美国内亚洲| 欧美亚韩一区| 国产精品成人v| 正在播放欧美一区| 亚洲视频在线一区观看| 蜜臀av一级做a爰片久久| 久久狠狠亚洲综合| 久久久噜噜噜久久人人看| 亚洲高清av| 一本色道久久88综合日韩精品| 免费亚洲电影在线观看| 一二三区精品| 羞羞视频在线观看欧美| 久久久999| 亚洲一区二区精品在线| 亚洲高清三级视频| 黄色成人91| 激情久久久久久| 国产精品久久久久久一区二区三区 | 欧美精品国产| 影音先锋久久久| 亚洲图片你懂的| 国产精品区一区二区三区| 亚洲精品国产精品国产自| 亚洲精品一区中文| 欧美亚洲免费电影| 亚洲精品免费网站| 欧美国产日韩一区二区三区| 欧美中文字幕| 亚洲欧美国产77777| 一区二区三区国产在线观看| 亚洲经典三级| 精品99一区二区| 好看不卡的中文字幕| 国产精品一二三| 国产精品乱码妇女bbbb| 欧美日韩亚洲系列| 欧美日韩一区二区在线| 欧美日韩dvd在线观看| 欧美精品v日韩精品v国产精品 | 欧美在线精品免播放器视频| 亚洲午夜精品| 在线视频欧美一区| 一区二区三区国产精品| 夜夜嗨网站十八久久| 一本久久青青| 亚洲影视综合| 欧美专区一区二区三区| 久久精品视频免费观看| 久久精品综合| 久久在线免费观看| 男人的天堂亚洲| 欧美激情二区三区| 欧美日韩一区二区三区在线观看免| 欧美日韩精品高清| 欧美肉体xxxx裸体137大胆| 欧美色图五月天| 国产精品丝袜xxxxxxx| 国产综合色精品一区二区三区| 影院欧美亚洲| 一区二区三区产品免费精品久久75 |