實(shí)現(xiàn)一個不拋異常的swap函數(shù)

template<typename T>
void swap(T& a, T& b)
{
    T temp(a);  // 拷貝構(gòu)造
    a = b;      // 拷貝賦值運(yùn)算符
    b = temp;   // 拷貝賦值運(yùn)算符
}

一個基本的swap函數(shù)大概如上所述蘸际,但是對某些類型而言。典型比如:pimpl手法(pointer to implementation)徒扶,這些復(fù)制沒有必要粮彤。

class WidgetImpl
{
    public:
        ...
    private:
        int a, b, c;
        std::vector<double> v;    // 意味著拷貝開銷很大
};

class Widget
{
    public:
        Widget(const Widget& rhs);
        Widget& operator=(const Widget& rhs)
        {
            ...
            *pImpl = *(rhs.pImpl);
        }
    private:
        WidgetImpl* pImpl;
}

一旦需要置換兩個Widget對象值,我們只需要交換pImpl指針姜骡,但缺省的swap算法不知道這一點(diǎn)导坟,它不止拷貝三個Widget,還拷貝三個WidgetImpl對象圈澈。

我們希望能夠告訴std::swap惫周,當(dāng)Widgets被置換時真正該做的是置換其內(nèi)部的pImpl指針,確切的做法就是:將std::swap針對Widget特化康栈。

namespace std {
    template<>
    void swap<Widget>(Widget& a, Widget& b)
    {
        swap(a.pImpl, b.pImpl);
    }
}

通常我們不被允許改變std命名空間內(nèi)的任何東西递递,但可以為標(biāo)準(zhǔn)的template制造特化版本。
雖然上面的代碼看起來達(dá)到我們的目的谅将,但實(shí)際上這個函數(shù)無法通過編譯漾狼,因為它企圖訪問 a 和 b 內(nèi)的Impl指針,而那是private的饥臂。
但我們可以在內(nèi)部實(shí)現(xiàn)一個swappublic成員函數(shù)做真正的置換工作逊躁,然后將std::swap特化,令它調(diào)用成員函數(shù):

class Widget
{
    public:
        ...
        void swap(Widget& other)
        {
            using std::swap;
            swap(pImpl, other.pImpl);
        }
    ...
};

namespace std
{
    template<>
    void swap<Widget>(Widget& a, Widget &b)
    {
        a.swap(b);
    }
}

這種做法不僅能通過編譯隅熙,還與STL容器有一致性稽煤,因為所有的STL容器也都提供有public swap成員函數(shù)和std::swap特化版本。

  • using std::swap

  • 不拋異常
    成員版swap最好實(shí)現(xiàn)成不要拋異常囚戚,因為swap的一個最好應(yīng)用是幫助class提供異常安全保障酵熙。此技術(shù)基于一個假設(shè):成員版的swap絕不拋異常。但這一約束只約束于成員版驰坊,不可施行于非成員版匾二,因為swap缺省版本是以拷貝構(gòu)造和拷貝賦值操作符為基礎(chǔ),而一般情況下兩者都允許拋異常。

總結(jié)
  • 如果swap的缺省實(shí)現(xiàn)對你的classclass template提供可接受的效率察藐。那么不需要做額外的操作皮璧。
    如果swap的缺省實(shí)現(xiàn)版本效率不足,試著做下面的事情:
    • 提供一個public swap成員函數(shù)分飞,讓它高效的置換你的類型的兩個對象值悴务,并且這個swap函數(shù)絕不能拋出異常。
    • 在你的classtemplate所在的命名空間內(nèi)提供一個non-member swap譬猫,并令它調(diào)用上述swap成員函數(shù)讯檐。
    • 如果你在編寫一個class而非class template,為你的class特化std::swap染服。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末别洪,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子肌索,更是在濱河造成了極大的恐慌蕉拢,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,723評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件诚亚,死亡現(xiàn)場離奇詭異晕换,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)站宗,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,485評論 2 382
  • 文/潘曉璐 我一進(jìn)店門闸准,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人梢灭,你說我怎么就攤上這事夷家。” “怎么了敏释?”我有些...
    開封第一講書人閱讀 152,998評論 0 344
  • 文/不壞的土叔 我叫張陵库快,是天一觀的道長。 經(jīng)常有香客問我钥顽,道長义屏,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 55,323評論 1 279
  • 正文 為了忘掉前任蜂大,我火速辦了婚禮闽铐,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘奶浦。我一直安慰自己兄墅,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,355評論 5 374
  • 文/花漫 我一把揭開白布澳叉。 她就那樣靜靜地躺著隙咸,像睡著了一般沐悦。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上扎瓶,一...
    開封第一講書人閱讀 49,079評論 1 285
  • 那天所踊,我揣著相機(jī)與錄音,去河邊找鬼概荷。 笑死,一個胖子當(dāng)著我的面吹牛碌燕,可吹牛的內(nèi)容都是我干的误证。 我是一名探鬼主播,決...
    沈念sama閱讀 38,389評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼修壕,長吁一口氣:“原來是場噩夢啊……” “哼愈捅!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起慈鸠,我...
    開封第一講書人閱讀 37,019評論 0 259
  • 序言:老撾萬榮一對情侶失蹤蓝谨,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后青团,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體譬巫,經(jīng)...
    沈念sama閱讀 43,519評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,971評論 2 325
  • 正文 我和宋清朗相戀三年督笆,在試婚紗的時候發(fā)現(xiàn)自己被綠了芦昔。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,100評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡娃肿,死狀恐怖咕缎,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情料扰,我是刑警寧澤凭豪,帶...
    沈念sama閱讀 33,738評論 4 324
  • 正文 年R本政府宣布,位于F島的核電站晒杈,受9級特大地震影響嫂伞,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜桐智,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,293評論 3 307
  • 文/蒙蒙 一末早、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧说庭,春花似錦然磷、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,289評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽寡润。三九已至,卻和暖如春舅柜,著一層夾襖步出監(jiān)牢的瞬間梭纹,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,517評論 1 262
  • 我被黑心中介騙來泰國打工致份, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留变抽,地道東北人。 一個月前我還...
    沈念sama閱讀 45,547評論 2 354
  • 正文 我出身青樓氮块,卻偏偏與公主長得像绍载,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子滔蝉,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,834評論 2 345

推薦閱讀更多精彩內(nèi)容