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

曠野的呼聲

路漫漫其修遠兮 吾將上下而求索

常用鏈接

統計

最新評論

【原創】boost.shared_ptr的相等與等價

boost.shared_ptr的相等與等價

關于相等和不等

template<class T, class U> inline bool operator==(shared_ptr<T> const & a, shared_ptr<U> const & b)

{

    return a.get() == b.get();

}

 

template<class T, class U> inline bool operator!=(shared_ptr<T> const & a, shared_ptr<U> const & b)

{

    return a.get() != b.get();

}

由此可以看出,實際上它們就是簡單的比較內部原始指針是否相等。因此可以得出這樣的判斷,對于使用==來比較的時候,以下情況下ab會相等:

 

1.ab都為空或者是都包含空指針,或者是其中之一。比如說如果a為空,b為空指針,那么也是相等的。

2.如果ab存在著拷貝構造或者是賦值關系,那么它們也是相等的。

 

同時我們也應該注意到相等判斷只與內部對象的地址有關系和內部對象的值沒有關系,比如說:

shared_ptr<int> sp1 (new int (3));

    shared_ptr<int> sp2 (new int (3));

由于sp1sp2內部的指針肯定不相同,因此sp1sp2肯定不相等。

 

不等于比較簡單,就是!(a==b)

 

關于等價

對于STL的關聯容器,它們對于是否存在的判斷標準是等價而不是相等。而等價的定義是如此:

template <class Ty1, class Ty2>

inline bool equiv(shared_ptr<Ty1> left,

                shared_ptr<Ty2> right)

{

    return !(left < right) && !(right < left);

}

對于shared_ptr來說,當且僅當如下情況時,這個函數才返回true,也就是等價關系成立:

 

1.  leftright都為空。即:leftright都使用默認構造函數構造。

2.  leftright控制相同的受控源。即:leftright存在著某一方從另外一方拷貝構造或者賦值的關系。

 

對于等價判斷實際上我們選擇的是shared_ptr的重載<,通過查看實現源碼可以發現最終發生比較關系的是shared_count的一個成員數據指針pi_sp_counted_base * pi_;

    friend inline bool operator<(shared_count const & a, shared_count const & b)

    {

        return std::less<sp_counted_base *>()( a.pi_, b.pi_ );

}

因此,重載<只有當pi_不相同的時候才有可能返回true。上面的語句通常情況下等于: return a.pi_ < b.pi_。通過參考上面的判別式!(left < right) && !(right < left)我們可以知道,除非a.pi_b.pi_相同,否則判別式!(left < right) && !(right < left)永遠不可能為true

 

下面就可以把重點集中在什么時候會導致a.pi_b.pi_相等了。

 

通過查看源碼我們可以發現當且僅當shared_count類對象使用默認構造的時候pi_才會被初始化為0,其它有參數的情況都會發生內存分配,返回值保存在pi_當中,先考察下面這幾個構造函數:

shared_count()

template<class Y> explicit shared_count( Y * p )

template<class P, class D> shared_count( P p, D d )

template<class P, class D, class A> shared_count( P p, D d, A a )

template<class Y>  explicit shared_count( std::auto_ptr<Y> & r )

這其中只有shared_count沒有對pi_進行內存分配,其它都分配了,而它們的參數均是直接由shared_ptr直接傳遞進來的。而默認構造shared_ptr的時候,shared_count亦使用默認構造。下面是shared_ptr的數據成員:

    T * px;                     // contained pointer

boost::detail::shared_count pn;    // reference counter

 

下面的代碼證明了我所說的構造方式:

shared_ptr(): px(0), pn() // never throws in 1.30+

    {

    }

 

    template<class Y>

    explicit shared_ptr( Y * p ): px( p ), pn( p ) // Y must be complete

    {

        boost::detail::sp_enable_shared_from_this( pn, p, p );

    }

 

    //

    // Requirements: D's copy constructor must not throw

    //

    // shared_ptr will release p by calling d(p)

    //

 

    template<class Y, class D> shared_ptr(Y * p, D d): px(p), pn(p, d)

    {

        boost::detail::sp_enable_shared_from_this( pn, p, p );

    }

 

    // As above, but with allocator. A's copy constructor shall not throw.

 

    template<class Y, class D, class A> shared_ptr( Y * p, D d, A a ): px( p ), pn( p, d, a )

    {

        boost::detail::sp_enable_shared_from_this( pn, p, p );

}

 

上面這些代碼證明了一點,在沒有發生拷貝構造和賦值的前提下,只有當使用默認構造的shared_ptr才等價,其它情況都不等價,哪怕T * px;的這個px相等,因為它跟等價比較沒有任何關系。同時通過源碼分析我們也可以發現一個奇怪的現象,即:

如果使用的是默認構造,那么即便是shared_ptr的類型不同,那么它們也會等價。而對于相等判斷來說,這會導致一個編譯錯誤。

當然,你明白原理了就不奇怪了,因為你所比較的對象是類型無關的。下面的代碼將證明我所說的:

 

#include <iostream>

#include <boost/tr1/memory.hpp>

 

using namespace std;

using namespace std::tr1;

 

#define PrintExp( exp ) \

    cout<< boolalpha << "( " << # exp << " ) = " << (exp) << endl

 

template <class Ty1, class Ty2>

inline bool equiv(shared_ptr<Ty1> left,

                shared_ptr<Ty2> right)

{

    return !(left < right) && !(right < left);

}

 

int main()

{

    shared_ptr<int> sp1( new int(100) );

 

    shared_ptr<int> sp2;

 

    PrintExp( equiv( sp1, sp2 ) );

 

    shared_ptr<double> sp3;

 

    PrintExp( equiv( sp2, sp3 ) );

 

    shared_ptr<int> sp4;

 

    PrintExp( equiv( sp2, sp4 ) );

 

    shared_ptr<int> sp5( (int*)NULL );

 

    PrintExp( equiv( sp4, sp5 ) );

 

    return 0;

}

 

等價還有另外一種情況,即通過拷貝構造或者賦值,下面的代碼清楚的說明了為什么這些情況下,我們的pi_會相等最終導致判別式!(left < right) && !(right < left)為真:

 

    shared_count(shared_count const & r): pi_(r.pi_) // nothrow

#if defined(BOOST_SP_ENABLE_DEBUG_HOOKS)

        , id_(shared_count_id)

#endif

    {

        if( pi_ != 0 ) pi_->add_ref_copy();

    }

 

// throws bad_weak_ptr when r.use_count() == 0

explicit shared_count(weak_count const & r);

// constructs an empty *this when r.use_count() == 0

    shared_count( weak_count const & r, sp_nothrow_tag );

    shared_count & operator= (shared_count const & r) // nothrow

    {

        sp_counted_base * tmp = r.pi_;

 

        if( tmp != pi_ )

        {

            if( tmp != 0 ) tmp->add_ref_copy();

            if( pi_ != 0 ) pi_->release();

            pi_ = tmp;

        }  // 如果這里還有個else的話,那么就意味著pi_=tmp,因為tmp=r.pi_,則pi_=r.pi_

// 那么pi_=tmp;這個賦值動作有

// 沒有又有什么關系呢?

 

        return *this;

}

 

到這里我想我們對此應該形成了一個比較清楚的概念了吧,呵呵。由于!(left < right) && !(right < left)的支持,使得shared_ptr可以作為容器的Key,比如說setmap等。另外補充一下就是weak_ptr也有類似的行為。

 

 

下面再附上一篇,這選自《Effective STL》的條款19,對我們理解這些很有幫助。

 

條款19:了解相等和等價的區別

 

STL充滿了比較對象是否有同樣的值。比如,當你用find來定位區間中第一個有特定值的對象的位置,find必須可以比較兩個對象,看看一個的值是否與另一個相等。同樣,當你嘗試向set中插入一個新元素時,set::insert必須可以判斷那個元素的值是否已經在set中了。

 

find算法和setinsert成員函數是很多必須判斷兩個值是否相同的函數的代表。但它們以不同的方式完成,find對“相同”的定義是相等,基于operator==set::insert對“相同”的定義是等價,通常基于operator<。因為有定義不同,所以有可能一個定義規定了兩個對象有相同的值而另一個定義判定它們沒有。結果,如果你想有效使用STL,那么你必須明白相等和等價的區別。

 

操作上來說,相等的概念是基于operator==的。如果表達式“x == y”返回truexy有相等的值,否則它們沒有。這很直截了當,但要牢牢記住,因為xy有相等的值并不意味著所有它們的成員有相等的值。比如,我們可能有一個內部記錄了最后一次訪問的Widget類。

 

class Widget {

public:

   ...

 

private:

   TimeStamp lastAccessed;

   ...

};

我們可以有一個用于Widget的忽略這個域的operator==

 

bool operator==(const Widget& lhs, const Widget& rhs) {

   // 忽略lastAccessed域的代碼

}

在這里,兩個Widget即使它們的lastAccessed域不同也可以有相等的值。

 

等價是基于在一個有序區間中對象值的相對位置。等價一般在每種標準關聯容器(比如,setmultisetmapmultimap)的一部分——排序順序方面有意義。兩個對象xy如果在關聯容器c的排序順序中沒有哪個排在另一個之前,那么它們關于c使用的排序順序有等價的值。這聽起來很復雜,但實際上,它不。考慮一下,舉一個例子,一個set<Widget> s。兩個Widget w1w2,如果在s的排序順序中沒有哪個在另一個之前,那么關于s它們有等價的值。set<Widget>的默認比較函數是less<Widget>,而默認的less<Widget>簡單地對Widget調用operator<,所以w1w2關于operator<有等價的值如果下面表達式為真:

 

!(w1 < w2)            // w1 < w2時它非真

&&                     // 而且

!(w2<w1)              // w2 < w1時它非真

這個有意義:兩個值如果沒有哪個在另一個之前(關于某個排序標準),那么它們等價(按照那個標準)。

 

在一般情況下,用于關聯容器的比較函數不是operator<或甚至less,它是用戶定義的判斷式。(關于判斷式的更多信息參見條款39。)每個標準關聯容器通過它的key_comp成員函數來訪問排序判斷式,所以如果下式求值為真,兩個對象xy關于一個關聯容器c的排序標準有等價的值:

 

!c.key_comp()(x, y) && !c.key_comp()(y, x)    // c的排序順序中

                  // 如果xy之前它非真,

                  // 同時在c的排序順序中

                  // 如果yx之前它非真

表達式!c.key_comp()(x, y)看起來很丑陋,但一旦你知道c.key_comp()返回一個函數(或一個函數對象),丑陋就消散了。!c.key_comp()(x, y)只不過是調用key_comp返回的函數(或函數對象),并把xy作為實參。然后對結果取反,c.key_comp()(x, y)僅當在c的排序順序中xy之前時返回真,所以!c.key_comp()(x, y)僅當在c的排序順序中x不在y之前時為真。

 

要完全領會相等和等價的含義,考慮一個忽略大小寫的set<string>,也就是set的比較函數忽略字符串中字符大小寫的set<string>。這樣的比較函數會認為“STL”和“stL”是等價的。條款35演示了怎么實現一個函數,ciStringCompare,它進行了忽略大小寫比較,但set要一個比較函數的類型,不是真的函數。要天平這個鴻溝,我們寫一個operator()調用了ciStringCompare的仿函數類:

 

struct CIStringCompare:              // 用于忽略大小寫

   public                // 字符串比較的類;

   binary_function<string, string, bool> {     // 關于這個基類的信息

                      // 參見條款40

   bool operator()(const string& lhs,

         const string& rhs) const

   {

      return ciStringCompare(lhs, rhs);    // 關于ciStringCompare

   }                  // 是怎么實現的參見條款35

}

給定CIStringCompare,要建立一個忽略大小寫的set<string>就很簡單了:

 

set<string, CIStringCompare> ciss;      // ciss = “case-insensitive

               // string set”

如果我們向這個set中插入“Persephone”和“persephone”,只有第一個字符串加入了,因為第二個等價于第一個:

 

ciss.insert("Persephone");     // 一個新元素添加到set

ciss.insert("persephone");     // 沒有新元素添加到set中如果我們現在使用setfind成員函數搜索字符串“persephone”,搜索會成功,

 

if (ciss.find("persephone") != ciss.end())...     // 這個測試會成功但如果我們用非成員的find算法,搜索會失敗:

 

if (find(ciss.begin(), ciss.end(),

      "persephone") != ciss.end())...      // 這個測試會失敗那是因為“persephone”等價于“Persephone”(關于比較仿函數CIStringCompare),但不等于它(因為string("persephone") != string("Persephone"))。這個例子演示了為什么你應該跟隨條款44的建議優先選擇成員函數(就像set::find)而不是非成員兄弟(就像find)的一個理由。

 

你可能會奇怪為什么標準關聯容器是基于等價而不是相等。畢竟,大多數程序員對相等有感覺而缺乏等價的感覺。(如果不是這樣,那就不需要本條款了。)答案乍看起來很簡單,但你看得越近,就會發現越多問題。

 

標準關聯容器保持有序,所以每個容器必須有一個定義了怎么保持東西有序的比較函數(默認是less)。等價是根據這個比較函數定義的,所以標準關聯容器的用戶只需要為他們要使用的任意容器指定一個比較函數(決定排序順序的那個)。如果關聯容器使用相等來決定兩個對象是否有相同的值,那么每個關聯容器就需要,除了它用于排序的比較函數,還需要一個用于判斷兩個值是否相等的比較函數。(默認的,這個比較函數大概應該是equal_to,但有趣的是equal_to從沒有在STL中用做默認比較函數。當在STL中需要相等時,習慣是簡單地直接調用operator==。比如,這是非成員find算法所作的。)

 

讓我們假設我們有一個類似setSTL容器叫做set2CF,“set with two comparison functions”。第一個比較函數用來決定set的排序順序,第二個用來決定是否兩個對象有相同的值。現在考慮這個set2CF

 

set2CF<string, CIStringCompare, equal_to<string> > s; 在這里,s內部排序它的字符串時不考慮大小寫,等價標準直覺上是這樣:如果兩個字符串中一個等于另一個,那么它們有相同的值。讓我們向s中插入哈迪斯強娶的新娘(Persephone)的兩個拼寫:

 

s.insert("Persephone");

s.insert("persephone");

著該怎么辦?如果我們說"Persephone" != "persephone"然后兩個都插入s,它們應該是什么順序?記住排序函數不能分別告訴它們。我們可以以任意順序插入,因此放棄以確定的順序遍歷set內容的能力嗎?(不能已確定的順序遍歷關聯容器元素已經折磨著multisetmultimap了,因為標準沒有規定等價的值(對于multiset)或鍵(對于multimap)的相對順序。)或者我們堅持s的內容的一個確定順序并忽略第二次插入的嘗試(“persephone”的那個)? 如果我們那么做,這里會發生什么?

 

if (s.find("persephone") != s.end())... // 這個測試成功或失敗?大概find使用了等價檢查,但如果我們為了維護s中元素的一個確定順序而忽略了第二個insert的調用,這個find會失敗,即使“persephone”的插入由于它是一個重復的值的原則而被忽略!

 

總之,通過只使用一個比較函數并使用等價作為兩個值“相等”的意義的仲裁者,標準關聯容器避開了很多會由允許兩個比較函數而引發的困難。一開始行為可能看起來有些奇怪(特別是當你發現成員和非成員find可能返回不同結果),但最后,它避免了會由在標準關聯容器中混用相等和等價造成的混亂。

 

有趣的是,一旦你離開有序的關聯容器的領域,情況就變了,相等對等價的問題會——已經——重臨了。有兩個基于散列表的非標準(但很常見)關聯容器的一般設計。一個設計是基于相等,而另一個是基于等價。我鼓勵你轉到條款25去學更多這樣的容器和設計以決定該用哪個。

 

posted on 2009-05-18 11:12 董波 閱讀(2313) 評論(0)  編輯 收藏 引用


只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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热福利| 亚洲人成精品久久久久| 国产视频一区免费看| 国产视频精品va久久久久久| 欧美不卡三区| 欧美高清视频在线| 亚洲国内欧美| 亚洲一级二级| 久久精品国产77777蜜臀| 久久精品国产视频| 欧美精品二区三区四区免费看视频| 欧美视频在线看| 国产精品久久婷婷六月丁香| 国产老女人精品毛片久久| 国产日本欧美一区二区| 亚洲国产美女| 亚洲福利一区| 久久精品国产91精品亚洲| 欧美大成色www永久网站婷| 亚洲人体偷拍| 蜜臀av在线播放一区二区三区| 老色鬼久久亚洲一区二区| 亚洲精品国久久99热| 欧美专区在线观看一区| 欧美成人在线免费观看| 欧美精品一区三区在线观看| 国产一区二区三区久久| 亚洲婷婷在线| 亚洲精品1234| 免费久久久一本精品久久区| 国产午夜精品一区二区三区欧美 | 国产精品h在线观看| 国产精品一区=区| 免费黄网站欧美| 在线成人av.com| 免费观看久久久4p| 麻豆精品视频| 日韩一级精品| 亚洲欧美激情四射在线日| 国产精品无码永久免费888| 午夜欧美不卡精品aaaaa| 欧美一区二区黄| 91久久精品久久国产性色也91| 国产精品99一区| 亚洲一区尤物| 久久精品二区亚洲w码| 亚洲丰满在线| 欧美成人三级在线| 欧美精品一区二区三| 欧美一级理论片| 欧美阿v一级看视频| 亚洲欧美福利一区二区| 久久久久久久久久久久久女国产乱 | 性色av一区二区三区红粉影视| 亚洲激情在线播放| 欧美日韩免费高清一区色橹橹| 亚洲欧美日韩一区| 久久深夜福利免费观看| 宅男66日本亚洲欧美视频| 久久高清一区| 久久精品国产久精国产爱| 欧美色大人视频| 欧美激情一区二区久久久| 国产亚洲精品aa| 日韩一级在线观看| 久久久精品日韩| 亚洲国产综合在线| 久久国产精品99精品国产| 亚洲一区二区影院| 欧美日韩xxxxx| 亚洲老司机av| 一区二区精品在线| 欧美视频在线观看免费| 一本色道久久综合亚洲精品不| 亚洲人精品午夜| 欧美日韩一区在线观看视频| 亚洲乱码国产乱码精品精天堂| 亚洲人成在线影院| 国产精品久久网站| 久久精品欧美日韩| 欧美激情网友自拍| 亚洲欧美综合v| 国产精品一区二区三区四区| 欧美一进一出视频| 免费久久精品视频| 亚洲少妇中出一区| 一区免费观看| 国产精品mv在线观看| 亚洲一区在线免费| 欧美高清在线播放| 亚洲欧洲av一区二区| 国内综合精品午夜久久资源| 欧美大片免费久久精品三p| 亚洲日本乱码在线观看| 亚洲欧美在线播放| 亚洲精品一区二区三| 国产日韩欧美一区二区三区四区| 蜜桃精品一区二区三区| 亚洲一区综合| 亚洲六月丁香色婷婷综合久久| 欧美在线三级| 亚洲免费影视第一页| 欧美不卡高清| 亚洲欧美在线看| 99re6热在线精品视频播放速度| 久久久999| 久久精品日韩| 欧美一区二区三区在线观看| 亚洲一区二区综合| 亚洲最新中文字幕| 日韩一本二本av| 亚洲精品一区二区三区蜜桃久| 国产一区在线观看视频| 国产日韩精品一区二区| 国产精品青草久久| 国产精品一区二区三区四区| 国产精品高潮呻吟久久| 久久久久亚洲综合| 欧美黄色一区| 国产精品jizz在线观看美国| 欧美日韩中文字幕精品| 国产精品久久久久久福利一牛影视| 欧美日韩亚洲精品内裤| 欧美亚一区二区| 国产综合色在线视频区| 91久久嫩草影院一区二区| 亚洲毛片播放| 久久国产精品一区二区三区| 欧美高清视频一区二区三区在线观看 | 亚洲国产老妈| 日韩网站免费观看| 亚洲欧美日韩综合| 久久久五月婷婷| 欧美性大战xxxxx久久久| 国产精品亚洲成人| 亚洲日本电影在线| 欧美中文字幕在线视频| 欧美电影免费观看网站| 亚洲一区二区三区在线观看视频| 久久精品国语| 国产偷国产偷精品高清尤物| 亚洲国产黄色| 国产精品白丝av嫩草影院 | 久久频这里精品99香蕉| 国产精品久久一级| 亚洲精品之草原avav久久| 久久久久九九视频| 亚洲视频专区在线| 欧美日韩一区二区在线观看视频| 国内精品一区二区三区| 欧美一区二区三区四区在线| 亚洲欧洲在线免费| 免费欧美高清视频| 亚洲国产视频一区二区| 久久精品欧洲| 久久精品99无色码中文字幕| 国产精品普通话对白| 性刺激综合网| 久久国产精品黑丝| 一区在线电影| 亚洲韩国精品一区| 欧美日韩国产欧| 亚洲欧美经典视频| 亚洲欧美日本精品| 极品尤物av久久免费看| 欧美激情精品久久久久| 欧美美女视频| 欧美在线综合| 免费黄网站欧美| 亚洲欧美清纯在线制服| 久久精品最新地址| 亚洲精品女人| 亚洲欧美变态国产另类| 亚洲国产另类 国产精品国产免费| 亚洲国产精品久久久久婷婷老年 | 久久九九热re6这里有精品 | 日韩午夜一区| 亚洲综合色自拍一区| 欧美在线一二三区| 亚洲综合日韩在线| 久久躁日日躁aaaaxxxx| 亚欧成人精品| 欧美色视频日本高清在线观看| 久久久久久尹人网香蕉| 欧美色道久久88综合亚洲精品| 久久久久国内| 国产女人18毛片水18精品| 中文网丁香综合网| 在线亚洲精品福利网址导航| 久久亚洲午夜电影| 另类激情亚洲| 黄色精品一区二区| 久久精品日韩欧美| 久久婷婷麻豆| 亚洲福利精品| 欧美高潮视频| 日韩一级在线| 欧美尤物一区| 亚洲第一在线综合网站|