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

牽著老婆滿街逛

嚴以律己,寬以待人. 三思而后行.
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>
            国产亚洲一区二区三区在线观看| 麻豆精品传媒视频| 国产精品99一区二区| 久久国产88| 亚洲一区中文| 你懂的国产精品永久在线| 午夜精品亚洲| 午夜精品剧场| 亚洲一区二区三区视频播放| 日韩小视频在线观看| 亚洲日本va午夜在线电影| 欧美日韩精品不卡| 欧美激情二区三区| 男男成人高潮片免费网站| 久久免费高清视频| 久久久久久999| 久久在线视频在线| 久久综合一区| 久久精品国产在热久久| 久久久九九九九| 另类激情亚洲| 欧美片在线播放| 欧美视频你懂的| 国产目拍亚洲精品99久久精品| 国产精品国产三级国产aⅴ9色| 国产精品久久久久久久久久久久| 欧美天堂亚洲电影院在线播放| 欧美人与性动交cc0o| 欧美日韩国产123区| 欧美日韩午夜在线| 国产精品白丝jk黑袜喷水| 国产精品亚洲精品| 国产日韩专区在线| 在线色欧美三级视频| 亚洲人成7777| 亚洲欧美成人综合| 在线视频一区二区| 欧美一区二区在线视频| 久久影音先锋| 亚洲精品久久久久久下一站| 99国产精品一区| 亚洲午夜一级| 蜜臀av一级做a爰片久久| 欧美日韩天天操| 国产日韩欧美综合一区| 西瓜成人精品人成网站| 久久久久久久999精品视频| 老妇喷水一区二区三区| 欧美日韩午夜视频在线观看| 国产日韩免费| 中文高清一区| 久久久夜精品| 日韩视频中午一区| 久久免费少妇高潮久久精品99| 欧美日韩视频在线一区二区 | 最新亚洲视频| 欧美中文字幕久久| 欧美日韩久久久久久| 国产欧美日韩亚洲一区二区三区| 国产麻豆一精品一av一免费| 99国产精品久久久| 久久综合一区二区| 一区二区三区视频观看| 久久综合激情| 国产一区二区三区四区老人| aa亚洲婷婷| 欧美在线观看网站| 一本一本久久a久久精品综合麻豆| 久久久久久久网站| 国产亚洲精品福利| 午夜精品久久一牛影视| 99国产精品| 欧美激情第六页| 91久久精品一区| 免费国产自线拍一欧美视频| 亚洲第一页在线| 免费久久久一本精品久久区| 国产日韩欧美| 午夜亚洲福利在线老司机| 亚洲人成网站777色婷婷| 亚洲欧美国产一区二区三区| 欧美日韩高清不卡| 夜夜狂射影院欧美极品| 亚洲高清免费视频| 欧美国产一区视频在线观看| 亚洲精品1区2区| 亚洲一区二区在线看| 亚洲美女av黄| 欧美日韩一区二区在线| 亚洲精品免费一二三区| 欧美黄污视频| 久久国产主播精品| 好看的av在线不卡观看| 久久久亚洲欧洲日产国码αv| 亚洲专区国产精品| 国产情人综合久久777777| 性色一区二区| 99精品视频免费观看视频| 欧美日韩精品一本二本三本| 男女激情视频一区| 久久婷婷国产综合精品青草| 日韩视频免费看| 欧美日韩在线影院| 午夜精品免费在线| 欧美一区在线视频| 在线看欧美视频| 亚洲国产精品99久久久久久久久| 亚洲欧美日韩国产一区二区三区| 国产日韩欧美日韩大片| 欧美在线短视频| 亚洲国产精品va在线看黑人| 亚洲一区自拍| 亚洲美女性视频| 久久这里只精品最新地址| 午夜精品免费视频| 欧美精品一级| 欧美福利电影网| 国产亚洲精品aa午夜观看| 99精品热视频| 99国产精品久久久久老师| 欧美一区深夜视频| 亚洲一区二区三区免费在线观看| 裸体歌舞表演一区二区| 久久九九全国免费精品观看| 欧美日韩在线视频一区二区| 欧美激情一区二区三区在线视频| 国产亚洲在线| 亚洲欧美精品suv| 亚洲主播在线观看| 欧美日韩国内| 亚洲美女色禁图| 亚洲免费不卡| 欧美激情91| 亚洲国产你懂的| 亚洲毛片av在线| 欧美sm视频| 亚洲高清视频在线| 亚洲第一主播视频| 久久这里只有| 欧美国产日本| 亚洲美女淫视频| 欧美精品久久一区二区| 亚洲国产欧美不卡在线观看| 亚洲国产精品久久久久婷婷884 | 亚洲一区在线观看视频| 欧美精品在线看| 亚洲精品一区二| 一区二区三区欧美激情| 欧美日韩一区高清| 一区二区三区欧美在线观看| 亚洲一区综合| 国产精品日韩欧美一区二区| 亚洲在线播放| 久久亚洲影音av资源网| 亚洲二区在线| 欧美日韩一区二区三区四区在线观看| 亚洲精品一二三| 欧美一级视频| 亚洲电影在线| 欧美日韩妖精视频| 午夜精品成人在线视频| 美女日韩欧美| 日韩午夜黄色| 国产精品日韩一区| 蘑菇福利视频一区播放| 久久精品视频在线观看| 一区二区三区中文在线观看 | 久久久午夜视频| 亚洲国产一区二区三区高清| 亚洲一二三区精品| 国产一二三精品| 欧美国产在线视频| 亚洲综合色激情五月| 免费观看成人www动漫视频| 在线视频免费在线观看一区二区| 国产乱人伦精品一区二区| 久热爱精品视频线路一| 一本久道久久综合婷婷鲸鱼| 久久久亚洲影院你懂的| 在线一区二区日韩| 激情久久影院| 国产精品久久久久aaaa| 久热精品在线| 欧美亚洲在线视频| 亚洲蜜桃精久久久久久久| 久久免费视频观看| 亚洲一区二区三区视频| 最新日韩欧美| 国产一区二区精品在线观看| 欧美精品激情| 久久五月天婷婷| 午夜精品免费| 亚洲视频一二区| 亚洲国产日韩欧美一区二区三区| 午夜精品美女久久久久av福利| 亚洲精品视频一区二区三区| 精品动漫av| 国产在线精品成人一区二区三区| 欧美午夜寂寞影院| 欧美精品aa|