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

牽著老婆滿街逛

嚴以律己,寬以待人. 三思而后行.
GMail/GTalk: yanglinbo#google.com;
MSN/Email: tx7do#yahoo.com.cn;
QQ: 3 0 3 3 9 6 9 2 0 .

引用計數我不怕之智能指針

轉載自:http://www.alisdn.com/wordpress/?p=415
作者:陳海



前言

使用引記數,就算是再歷害的高手也難免會出錯。而一但出錯了,之后再去查問題可就相當的困難了。正如我曾經看到,有一段代碼是這樣的:

m_spView->Release();
m_spView
->Release();
m_spView
->Release();

 

看到這段代碼,就知道引用計數出問題了。他想通過這種方式,把多出來的計數Release掉。但這么做能解決問題嗎?答案是不能,這樣的代碼還可能造成嚴重的穩定性問題。解決引用計數問題,除了要了解引用計數規則外,我們還提昌要用智能指針。智能能幫助我們很好的處理引用計數問題。

 

智能指針的差異

在用VC開發應用程序時,有兩個引用計數類可供我們使用。_com_ptr_t與CComPtr,它們都能很好的幫助我們解決引用計數處理。但這兩個類還是有一點小小的區別,有的時候這一點區別也是致命的,因此我們必須清楚它們的差別。下面我羅列了它們之間的差別:

  1. CComPtr的&運算符不會釋放原指針,而_com_ptr_t會釋放原指針。
  2. CComPtr對AddRef與Release做了限制,也就是不充許調用這兩個方法,而_com_ptr_t并沒有限制。
  3. CComPtr只能接受模版參數指定的指針,_com_ptr_t可以接受任何接口指針,并自動調用QueryInterface得到模板參數指定的指針類型。

這些區別,導致了有些代碼不能同時應用于兩個智能指針。

 

&運算符差異帶來的風險

HRESULT hr GetView(int i, /*out*/IView** ppView)
{
    
*ppView = m_Views[i];
    (
*ppView)->AddRef();


return S_OK;
}

}


CComPtr
<IView> spView;
for (int i = 0; i < 10; i++)
{
    GetView(i, 
&spView);
spView
->

以上代碼會導致引用計數出錯,前面的9個View的引用計數并沒有Release。CComPtr<IView>的&運算符,會返回IView**也就是CComPtr內部成員的地址,但它不釋放原來的指針。而GetView又會修改指針,直接把原來的指針拋棄了。

這個代碼可以這樣改:

for (int i = 0; i < 10; i++)
{
    CComPtr
<IView> spView;
    GetView(i, 
&spView);
    spView
->
}

把指針作為循環的局部變量,這樣每次循環退出前spView都會被析構,最終調用Release。當然還能這樣改:

COM_SMARTPTR_TYPEDEF(IView, __uuidof(IView));
IView Ptr spView;
for (int i = 0; i < 10; i++)
{
    GetView(i, 
&spView);
    spView
->
}

_com_ptr_t的&運算符會幫助我們把原來的指針Release掉,所以我們就不必擔心引用計數沒有釋放。

 

禁用AddRef與Release

然我們使用的智能指針,就不要再去調AddRef或Release了。如果再去手工調用它們,就失去了智能指針的好處。CComPtr有一個非常巧妙的方法,禁止調用這兩個方法。它聲明了一個類_NoAddRefReleaseOnCComPtr,它的定義如下:

template <class T>
class _NoAddRefReleaseOnCComPtr : public T
{
    
private:
        STDMETHOD_(ULONG, AddRef)()
=0;
        STDMETHOD_(ULONG, Release)()
=0;
}
;

我們看到,里面就定義了兩個私有函數。AddRef與Release,它們重寫了IUnknown的這兩個方法,并且繼承自模板T。再來看段代碼:

 

_NoAddRefReleaseOnCComPtr<T>* operator->() const
throw()
{
    ATLASSERT(p
!=NULL);
    
return (_NoAddRefReleaseOnCComPtr<T>*)p;
}

我們看到的是CComPtr的”->”運算符,它將內部的指針強制轉換成_NoAddRefReleaseOnCComPtr<T>*。其中T是CComPtr的模板參數,也就是接口指針類型??梢钥闯鯻NoAddRefReleaseOnCComPtr<T>繼承自接口類型,因此通過_NoAddRefReleaseOnCComPtr<T>*可以調用T的所有函數。前面我們看到NoAddRefReleaseOnCComPtr的兩個私用函數,AddRef與Release,如果有誰想調用就會報編譯錯誤。

 

自動QueryInterface

_com_ptr_t有多個=運算符版本,代碼如下:

template<typename _IIID> class _com_ptr_t
{
public:
typedef _IIID ThisIIID;
typedef
typename _IIID::Interface Interface;

// Queries for interface.

template
<typename _InterfaceType> _com_ptr_t& operator=(_InterfaceType* p)
{
HRESULT hr 
= _QueryInterface(p);

if (FAILED(hr) && (hr != E_NOINTERFACE)) {
_com_issue_error(hr);
}


return *this;
}


// Saves the interface.

template
<> _com_ptr_t& operator=(Interface* pInterface) throw()
{
if (m_pInterface != pInterface) {
Interface
* pOldInterface = m_pInterface;
m_pInterface 
= pInterface;
_AddRef();

if (pOldInterface != NULL) {
pOldInterface
->Release();
}

}

return *this;
}

其中
template<typename _InterfaceType> _com_ptr_t& operator=(_InterfaceType* p)是一個模板函數,接受任意類型的指針,函數內部會調用傳入參數”p”的QueryInterface。

template<> _com_ptr_t& operator=(Interface* pInterface) throw()是模板函數的一個偏特化版本,接受_com_ptr_t模板參數中指定的指針類型。當傳入的接口指針類型,與類模板指定的類型一樣時這個函數會被調用。它不需要做QueryInterface的調用,只是簡單的AddRef;

綜上所述,兩個智能指針在同一份代碼里混用,很可能導致不良后果。所以我認為,最好不要在同一份代碼里混用。而這兩個指針,我很喜歡_com_ptr_t,它在許多方面明顯優于CComPtr。


Attach與Detach

使用了智能指針,也并不是高枕無憂了。它還是給我們帶來了一些新的問題。

有一些第三方類庫設計的不合理,它在函數的返回值里返回接口指針。如下代碼就導會導致引用計數泄漏:

 

IView* GetView(int nIndex)
{
    IView
* pView = m_Views[nIndex];

    pView
->AddRef();

    
return pView;
}


IViewPtr spView 
= GetView(0);

以上代碼,注意調用GetView的地方。IViewPtr是智能指針,它的=運算符是會再調用AddRef而GetView里已經調了一次AddRef了,這里多了一次AddRef。別問我GetView中為什么要AddRef,這是引用計數規則,不清楚請看《引用計數我不怕之引用計數規則》。

解決這個問題的方法就是用Attach函數

IViewPtr spView;

spView.
Attach(GetView(
0));

也許是有人寫了Attach,而其它人不明白Attach的意思,結果寫出了這樣的代碼。

void SetView(IView* pView)
{
    m_spView.Attach(pView);
}

根據引用計數規則,將指針保存為副本,必須AddRef。但是這個例子里沒有這么干,結果m_spView變成了野指針。

前面我們看到的GetView很簡單,但是下面我們要做一別的事情,于是要用智能指針。

HRESULT hr GetView(int nIndex, IView** ppView)
{
    IViewPtr spView 
= m_Views[nIndex];

    
if (spView->IsVisable() != S_OK)
        
return E_FAILD;

    
*ppView = spView;

    
return S_OK;
}

表面看來沒什么問題,但在函數返回后,智能指針又會調用一次Release。要解決這個問題,可以調用Detach。

HRESULT hr GetView(int nIndex, IView** ppView)
{
    IViewPtr spView 
= m_Views[nIndex];

    
if (spView->IsVisable() != S_OK)
        
return E_FAILD; 

    
*ppView = spView.
Detach();

    
return S_OK;
}

Detach還是會被亂用,看到這樣的代碼還真是哭笑不得。

 

HRESULT hr ChangeView(int nIndex)
{
    IViewPtr spView 
= m_Views[nIndex].Detach();

    spView
->Change();

    
return S_OK;
}

這段代碼能導致兩個問題

  1. 引用計數泄漏
  2. m_Views中的指針變成空了

泄漏是由于Detach返回IView*,并不會Release,而spView又會再調用一次AddRef。智能指針的Detach是會把自己設成空的,否則還叫什么Detach。

使用智能指針,是解決引用計數問題最好的辦法。不要因為用智能指針,會引入新的問題,而放棄使用它。只要花心思搞清楚智能指針的不同點,使用時注意一些細節問題,使用起來應該會變的非常輕松。


posted on 2011-01-18 16:14 楊粼波 閱讀(1518) 評論(0)  編輯 收藏 引用

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲一区在线免费| 亚洲精品在线一区二区| 欧美自拍偷拍午夜视频| 国产欧美日韩专区发布| 欧美欧美午夜aⅴ在线观看| 欧美午夜精品理论片a级按摩| 在线视频欧美日韩| 亚洲一区二区三区四区五区黄 | 乱人伦精品视频在线观看| 久久精品一区二区| 99re6热在线精品视频播放速度| 99视频精品在线| 国产乱码精品一区二区三区不卡| 久久精品在线视频| 欧美成人精品影院| 亚洲综合精品四区| 久久久高清一区二区三区| 亚洲免费观看高清在线观看 | 蜜桃久久av一区| 欧美日韩99| 久久精品视频免费| 欧美日韩精品三区| 久久综合久久久| 欧美日韩亚洲一区二区三区| 久久精品国产在热久久 | 亚洲天堂网在线观看| 樱花yy私人影院亚洲| 亚洲丝袜av一区| 亚洲激情视频在线| 欧美亚洲综合在线| 亚洲午夜精品国产| 久久久天天操| 欧美一级久久久| 欧美日韩国产精品一区二区亚洲| 久久久久欧美| 国产精品久久一区二区三区| 亚洲人成7777| 国一区二区在线观看| 在线视频你懂得一区| 亚洲精品久久嫩草网站秘色| 久久九九久精品国产免费直播| 亚洲午夜久久久| 欧美激情2020午夜免费观看| 久久综合图片| 国内精品视频666| 亚洲一二三区精品| 亚洲天堂男人| 欧美日本三区| 亚洲韩日在线| 亚洲国产精品一区制服丝袜| 久久精品人人做人人爽| 久久超碰97中文字幕| 国产精品嫩草影院av蜜臀| 亚洲精品日韩在线观看| 亚洲日本欧美| 欧美韩国在线| 亚洲免费观看| 在线视频亚洲一区| 欧美色道久久88综合亚洲精品| 亚洲电影在线播放| 亚洲第一在线视频| 久久影院亚洲| 亚洲第一福利在线观看| 亚洲黄色片网站| 欧美成年人视频网站欧美| 久久久亚洲国产天美传媒修理工| 欧美激情一区二区三区在线视频| 激情欧美一区| 免费观看在线综合| 亚洲国产日韩欧美综合久久| 亚洲精品国产精品国自产在线 | 亚洲一区在线看| 欧美伊人久久久久久午夜久久久久| 国产精品欧美久久| 午夜欧美精品| 美女视频黄a大片欧美| 91久久久久久国产精品| 欧美福利一区| 99国产精品久久久久久久成人热| 亚洲一区日韩在线| 国产一区二区av| 另类激情亚洲| 一本久道久久综合婷婷鲸鱼| 欧美在线短视频| 亚洲高清不卡在线| 欧美视频在线不卡| 香蕉成人伊视频在线观看| 蜜月aⅴ免费一区二区三区| 亚洲免费观看高清在线观看 | 在线天堂一区av电影| 国产精品人人做人人爽| 久久大逼视频| 亚洲人成在线播放| 久久福利一区| 亚洲精品久久7777| 国产日韩欧美精品| 欧美好骚综合网| 欧美在线免费| 99国内精品久久久久久久软件| 久久精品夜夜夜夜久久| 日韩午夜剧场| 国内精品久久久久久久影视蜜臀| 久久综合给合| 午夜国产精品影院在线观看| 亚洲大片在线| 久久亚洲欧洲| 亚洲一区欧美二区| 亚洲日本精品国产第一区| 国产老肥熟一区二区三区| 欧美a一区二区| 欧美专区在线观看| 亚洲香蕉在线观看| 亚洲福利视频在线| 久久躁日日躁aaaaxxxx| 午夜精品久久久久久久99热浪潮 | 久久av二区| 亚洲图片你懂的| 亚洲精品老司机| 好吊妞这里只有精品| 国产精品一二一区| 欧美日韩一区二区三| 美国三级日本三级久久99| 性色一区二区| 亚洲免费人成在线视频观看| 亚洲欧洲一区二区三区久久| 男人的天堂成人在线| 久久久久在线| 欧美在线视频一区二区三区| 亚洲永久精品国产| 国产欧美va欧美va香蕉在| 亚洲视频高清| 99国产麻豆精品| 亚洲精品在线视频| 亚洲精品久久7777| 亚洲精品国精品久久99热| 欧美大片在线看免费观看| 蜜桃av综合| 欧美顶级少妇做爰| 亚洲第一毛片| 亚洲七七久久综合桃花剧情介绍| 欧美大胆a视频| 欧美顶级少妇做爰| 91久久中文字幕| 亚洲人成网站在线播| 亚洲精品乱码| 亚洲视频免费看| 亚洲一区美女视频在线观看免费| 亚洲一区二区三区视频| 午夜伦欧美伦电影理论片| 午夜欧美理论片| 久久精品成人欧美大片古装| 久久久精品动漫| 美腿丝袜亚洲色图| 欧美激情第1页| 国产精品久在线观看| 国产日产精品一区二区三区四区的观看方式 | 久久亚洲高清| 亚洲大片免费看| 日韩视频―中文字幕| 在线视频日韩| 久久国产精品久久久| 久久综合久久综合久久综合| 欧美电影免费观看网站| 欧美亚日韩国产aⅴ精品中极品| 国产女主播一区二区| 亚洲国产成人不卡| 亚洲图片欧洲图片日韩av| 久久精品人人| 欧美激情自拍| 亚洲深夜福利| 久久综合伊人77777| 欧美久久久久久久久| 国产手机视频精品| 亚洲日本电影| 欧美主播一区二区三区美女 久久精品人| 久久一区亚洲| 一本久久a久久精品亚洲| 欧美一区二区三区久久精品| 欧美激情一二三区| 国产日产欧产精品推荐色| 亚洲国产日韩在线一区模特| 亚洲欧美日韩国产综合在线 | 亚洲午夜一级| 久久综合色播五月| 一区二区三区不卡视频在线观看 | 狼人天天伊人久久| 国产精品一区二区在线观看不卡 | 亚洲免费观看| 久热精品视频| 亚洲欧美一区二区视频| 欧美成人精品三级在线观看| 国产亚洲欧洲| 午夜精品久久久久久久99热浪潮 | 久久精品国产欧美激情| 欧美色欧美亚洲另类七区| 亚洲国产美女| 久久综合一区| 欧美影院午夜播放| 国产精品夜夜夜| 亚洲人成网站777色婷婷|