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

Error

C++博客 首頁 新隨筆 聯系 聚合 管理
  217 Posts :: 61 Stories :: 32 Comments :: 0 Trackbacks
首先一個原則是這樣:被用來delete的指針,一定是new出來的。
在設計智能指針的時候發現,如果采用delete this機制,到最后可能會有嚴重的問題。
假設有這樣的多重繼承:
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不一樣,這樣就出現堆錯誤。
如果設計入侵式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;
這樣的用法無法實現,應為沒有實現IXXX接口,是一個虛基類,實現了在調用的時候也是各種基類未指定的錯誤。

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



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這種,還是有部分場景更加格式,至少不用擔心類型轉換變得異常麻煩,,,
后發現如下實現方式:
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是預期額值,懷疑可能和編譯器有關,VS測試OK,gcc沒去實驗。
分析了一下原因:C++保證單根繼承的時候基類和派生類地址是一樣的,如果是多重繼承,那么也保證和最深的根父類地址樣,順便的,從最深的根節點,選一路下來是安全的。
class CRefTest : public TRefObj<CRefTest>所以約定TRefObj<CRefTest>這玩意和永遠和接口在一個級別,可以保證IRefObj永遠是最深的根,就安全了?
目前只在控件設計的時候用RefObj吧,,,坑的比較深,,,
posted on 2014-10-16 16:29 Enic 閱讀(784) 評論(2)  編輯 收藏 引用 所屬分類: C/C++技巧

評論

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

# re: C++多重繼承導致delete引起堆錯誤、智能指針設計陷阱 2014-10-30 14:19 Enic
用boost::shared_ptr 或者std::shared_ptr的時候沒有這個問題,我猜測是共享數據卡里邊保留了原始指針,,,@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>
            91久久精品国产91久久| 免费视频一区二区三区在线观看| 先锋影音国产精品| 久久综合久久久| 国产精品伦一区| 在线观看一区二区精品视频| 亚洲日本aⅴ片在线观看香蕉| 欧美凹凸一区二区三区视频| 亚洲欧美福利一区二区| 欧美成人激情在线| 韩日欧美一区二区三区| 亚洲欧美一区二区三区在线| 欧美aaaaaaaa牛牛影院| 欧美国产日本| 在线精品亚洲| 亚洲激情小视频| 蜜桃av综合| 中文欧美字幕免费| 欧美二区乱c少妇| 亚洲深夜影院| 日韩午夜精品视频| 欧美国产日本韩| 亚洲综合日韩在线| 夜夜狂射影院欧美极品| 欧美国产一区二区| 欧美亚洲一区| 亚洲综合国产激情另类一区| 激情五月婷婷综合| 免费一级欧美片在线播放| 嫩草成人www欧美| 亚洲伊人色欲综合网| 久久久精品2019中文字幕神马| 国产在线国偷精品产拍免费yy| 欧美一级久久久| 欧美激情国产精品| 久久亚洲视频| 卡通动漫国产精品| 午夜激情综合网| 欧美激情综合色| 亚洲图片在线观看| 久久综合国产精品台湾中文娱乐网| 一级成人国产| 亚洲一区二区三区免费观看| 国产精品视频专区| 久久久另类综合| 免费不卡中文字幕视频| 日韩西西人体444www| 日韩午夜一区| 亚洲精品影视| 亚洲性色视频| 好男人免费精品视频| 中文精品99久久国产香蕉| 亚洲国产清纯| 亚洲少妇一区| 亚洲午夜久久久久久久久电影院| 亚洲一区区二区| 日韩亚洲欧美成人| 免费美女久久99| 欧美大片一区二区三区| 久久精品国产99国产精品| 亚洲精品久久久久| 亚洲天堂网站在线观看视频| av不卡在线| 欧美日本成人| 久久综合色8888| 欧美色精品天天在线观看视频| 欧美主播一区二区三区美女 久久精品人 | 国产亚洲人成网站在线观看| 麻豆精品精华液| 狠狠色伊人亚洲综合网站色| 欧美一区二区免费观在线| 亚洲高清av| 亚洲中字黄色| 久久不见久久见免费视频1| 国产精品久久久久久久久| 欧美大学生性色视频| 亚洲第一精品久久忘忧草社区| 久久成年人视频| 欧美www在线| 亚洲乱码久久| 国产精品高潮久久| 午夜国产不卡在线观看视频| 日韩亚洲欧美一区| 欧美日韩视频免费播放| 欧美激情一区二区三区全黄| 国产欧美日韩一区二区三区在线| 日韩午夜av| 午夜久久资源| 一区在线影院| 99国内精品久久| 欧美一级片久久久久久久| 国产一区二区三区无遮挡| 久久免费偷拍视频| 久久夜色撩人精品| 亚洲免费观看高清在线观看 | 亚洲国产精品成人综合色在线婷婷| 久久蜜臀精品av| 亚洲精品在线一区二区| 亚洲高清久久| 欧美天堂在线观看| 久久久免费精品视频| 久久久人成影片一区二区三区| 亚洲国产精品v| 欧美视频二区| 久久婷婷久久一区二区三区| 亚洲乱码视频| 久久手机免费观看| 亚洲影院色无极综合| 1769国内精品视频在线播放| 欧美手机在线视频| 久久久精品国产一区二区三区| 亚洲精品久久久蜜桃 | 国产精品美女久久| 久久综合色播五月| 亚洲欧美韩国| 一区二区三区国产| 午夜亚洲影视| 日韩一二三在线视频播| 国产一区二区三区无遮挡| 欧美日韩一视频区二区| 麻豆久久婷婷| 久久精品天堂| 午夜精品一区二区在线观看| 日韩视频永久免费观看| 欧美韩国日本一区| 久久久久久9999| 亚洲第一页中文字幕| 国产日产精品一区二区三区四区的观看方式| 欧美88av| 免费在线播放第一区高清av| 久久国产精品久久久| 亚洲欧美一区二区视频| 亚洲私人影院在线观看| 一区二区三区波多野结衣在线观看| 亚洲七七久久综合桃花剧情介绍| 老司机免费视频一区二区| 久久久91精品国产一区二区精品| 亚洲中午字幕| 午夜精品久久久久久久99热浪潮| 日韩系列在线| 在线视频精品一区| 国产一区二区黄色| 国产午夜精品在线观看| 欧美高清不卡在线| 欧美成人午夜激情在线| 欧美成人激情视频| 欧美黄色大片网站| 欧美日本亚洲韩国国产| 欧美日韩中文字幕精品| 欧美亚男人的天堂| 麻豆成人在线观看| 蜜臀91精品一区二区三区| 免费久久久一本精品久久区| 蜜臀av在线播放一区二区三区| 久久综合狠狠综合久久激情| 欧美成人69av| 欧美三级电影一区| 国产精品扒开腿做爽爽爽软件| 国产精品免费区二区三区观看| 国产精品久久久久久久久久久久久 | 亚洲黄色在线| 99pao成人国产永久免费视频| 亚洲一区二区少妇| 欧美中文字幕不卡| 亚洲影院在线观看| 欧美专区在线观看| 狼人社综合社区| 欧美午夜精品久久久| 国产日产高清欧美一区二区三区| 一区二区三区亚洲| 99在线|亚洲一区二区| 午夜久久黄色| 欧美激情 亚洲a∨综合| 一本色道久久88综合日韩精品| 性欧美xxxx视频在线观看| 麻豆免费精品视频| 国产精品无码专区在线观看| 狠狠色狠狠色综合日日小说| 亚洲美女av网站| 久久福利影视| 亚洲精品美女免费| 香港成人在线视频| 欧美精品日韩一本| 黄色小说综合网站| 99国产精品自拍| 麻豆精品精华液| 亚洲午夜一二三区视频| 另类av导航| 国产日韩视频一区二区三区| 亚洲三级毛片| 久久久另类综合| 亚洲婷婷综合色高清在线| 美女视频网站黄色亚洲| 国产欧美精品xxxx另类| 日韩亚洲精品电影| 美女久久一区| 欧美一区免费视频| 国产精品毛片va一区二区三区| 亚洲精品1区| 久久一区二区三区国产精品 |