effective C++ 条款25 swap

item 25:一个不抛异常的swap函数

标准库有一个swap用于交换两个对象值

namespace std{

template<typename T>

void swap(T& a , T& b)

{

T temp(a) ;

a = b ;

b = temp ;

}

}

所以,只要对象的类型T支持copying(copy ctor 和 copy assignment),那么你就可以用这个标准库的swap,但是,你必须保证,你swap的时候,上述操作是你需要的,有的时候,没有必要,尤其是pimpl(pointer to implementation)手法。

例如下面这个例子:

class WidgetImpl{

public :

....

private :

int a , b , c  ;

....   //好多成员

};

class Widget

{

public:

Widget(const Widget & rhs) ;

Widget& operator=(const Widget & rhs)

{

*pImpl = *rhs.pImpl // 复制所指的对象

}

private:

WidgetImpl * pImpl ;

};

一旦要交换两个Widget,我们其实就让两个指针换一下就可以了,没有必要把指针所指的内容互换,那样效率太低。

想法1:

namespace std{

template<>  // 全特化

void swap<Widget> (Widget & a,  Widget & b)

{

swap(a.pImpl , b.pImpl) ;

}

}

上面这个想法编译不过,因为pImpl是private的,是不能被外界访问的。

想法2 : 加入类的成员函数swap,让模板函数去调用

class Widget{

public :

void swap(Widget & other)

{

using std::swap ;  // 后面讲

swap(p.Impl , other.pImpl) ;

}

...

};

namespace std{

template<>  // 全特化

void swap<Widget> (Widget & a,  Widget & b)

{

a.swap(b) ;

}

}

通过上面的写法,我们就能得到正确的,针对类别Widget 的swap函数。这就叫为你的class特化std::swap,并令它调用你的swap函数。

但是,问题,如果我现在是类模板,怎么办?

现在的类是:

template<typename T>

class WidgetImpl{...}

template<typename T>

class Widget{...}

如果你现在这么做:

namespace std{

template<typename T>  // 偏特化

void swap<Widget> (Widget<T> & a,  Widget<T> & b)

{

a.swap(b) ;

}

}

sorry不行了,原因有以下2点:

1、因为C++只允许对class template进行偏特化,不对能function template进行偏特化。

2、在std这个命名空间,不可以添加新的template到std里。上面,我们能加东西到std是因为,为标准的templates(eg : swap)制造特化版本。

为了解决上述2两个问题,引入下面的解决方法

想法3:转为class template量身打造

定义一个专门的命名空间,放入我们所有的东西

namespace WidgetStuff{

template<typename T>

class WidgetImpl{...}

template<typename T>

class Widget{...} // 这个里面有一个swap函数

template<tyoename T>

void swap(Widget<T>& a , Widget<T>& b)

{

a.swap(b) ;

}

}

我们在用swap函数的时候,应该这么用:

template<typename T>

void doSomething(T& obj1 , T& obj2)

{

using std::swap ;  // 这样即使没有T的专属版本的swap函数

swap(obj1 , obj2)  ;

}

上面就结束了我们swap函数的制作过程,那么为什么可行呢?这个都是利用了C++的名称查找规则

编译器用“实参取决之查找规则”,找出WidgetStuff中的swap函数,如果命名空间中没有T的专属swap函数,编译器就用std::swap,即使如此,编译器还是喜欢std::swap重T的专属版本,就是想法2会选中。所以,优先级顺序是想法3--》想法2--》标准库自己的swap

总结:

1、弄清楚什么时候我们需要自己写swap函数

2、在class 或者 class template提供一个public swap函数

3、在class 或者 class template的命名空间中提供一个non-member swap ,调用2中的swap

4、如果你正在编写一个class(不是template),为class特化std::swap。并令他调用2中的swap

5、当你实际用swap时,先using std::swap,然后不加任何修饰符,赤裸裸调用swap

成员版swap决不能抛出异常,因为很多异常安全性代码都是通过类的swap成员函数保证的,并且,我们自己去写swap,一般都是用指针,内置类型,这些类型不会抛异常,写swap是为了高效

时间: 2024-08-04 10:47:12

effective C++ 条款25 swap的相关文章

Effective C++ 条款25 考虑写出一个不抛出异常的swap函数

1. swap是STL的一部分,后来成为异常安全性编程(exception-safe programming)(见条款29)的一个重要脊柱,标准库的swap函数模板定义类似以下: namespace std{ template<typename T> swap(T& lhs,T& rhs){ T temp(lhs); lhs=rhs; rhs=temp; } } 只要T类型支持拷贝构造以及拷贝赋值,标准库swap函数就会调用T的拷贝构造函数和拷贝构造操作符完成值的转换,但对于某

Effective C++ -----条款25:考虑写出一个不抛异常的swap函数

当std::swap对你的类型效率不高时,提供一个swap成员函数,并确定这个函数不抛出异常. 如果你提供一个member swap,也该提供一个non-member swap用来调用前者.对于class(而非templates),也请特化std::swap. 调用swap时应针对std::swap使用using声明式,然后调用swap并且不带任何“命名空间资格修饰”. 为“用户定义类型”进行std templates全特化是好的,但千万不要尝试在std内加入某些对std而言全新的东西.

More Effective C++ 条款25 将constructor和non-member function虚化

1. Virtual constructor 原则上构造函数不能为虚:虚函数用于实现"因类型而异的行为",也就是根据指针或引用所绑定对象的动态类型而调用不同实体,但构造函数用于构造对象,在对象构造之前自然没有动态类型的概念,虚与非虚也就无从谈起.所谓的的virtual-constructor实际上是"仿virtual-constructor",它本质上不是constructor,但能够产生不同类型的对象,从而实现"virtual-constructor&q

Effective C++ 条款25

考虑写出一个不抛出异常的swap函数 本节讲解如何自定义一个高效的swap函数 对于std名空间中的swap缺省函数如下所示 namespace std{ template<typename T> void swap(T& a, T& b) { T temp(a); a=b; b=temp; } } class WidgetImpl{ public: -- private: int a,b,c; //数据很多,复制意味时间很长 std::vector<double>

条款25:考虑写出一个不抛异常的swap函数

条款25:考虑写出一个不抛异常的swap函数 swap函数在C++中是一个非常重要的函数,但实现也非常复杂. 看一个缺省的std::swap函数的实现 namespace std { template<typename T> void swap( T& a , T& b) { T temp(a); a = b; b = temp } } ①内置类型的调用 int a = 2; int b =3; std::swap(a, b); cout<<"a:&quo

Effective C++ Item 25 考虑写出一个不抛异常的swap函数

本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie 经验:当std::swap对你的类型效率不高时,提供一个swap成员函数,并确定这个函数不抛出异常 示例: stl里的swap算法 namespace std{ template<typename T> void swap(T &a, T &b){ T temp(a); a = b; b = temp; } } //"pimpl手法"(pointer

Effective C++ 条款11,12 在operator= 中处理&ldquo;自我赋值&rdquo; || 复制对象时不要忘记每一个成分

1.潜在的自我赋值     a[i] = a[j];     *px = *py; 当两个对象来自同一个继承体系时,他们甚至不需要声明为相同类型就可能造成别名. 现在担心的问题是:假如指向同一个对象,当其中一个对象被删,另一个也被删,这会造成不想要的结果. 该怎么办? 比如:   widget& widget:: operator+ (const widget& rhs) {    delete pd;    pd = new bitmap(*rhs.pb);    return *thi

More Effective C++ 条款35 让自己习惯于标准C++ 语言

(由于本书出版于1996年,因此当时的新特性现在来说可能已经习以为常,但现在重新了解反而会起到了解C++变迁的作用) 1. 1990年后C++的重要改变 1). 增加了新的语言特性:RTTI,namespaces,bool,关键词mutable和explicit,enums作为重载函数之自变量所引发的类型晋升转换,以及"在class 定义区内直接为整数型(intergral) const static class members设定初值"的能力. 2). 扩充了Templates的特性

effective c++ 条款4 make sure that objects are initialized before they are used

1 c++ 类的数据成员的初始化发生在构造函数前 class InitialData { public: int data1; int data2; InitialData(int a, int b) { data1 = a: //this is assignment data2 = b; //this is assignment } /* InitialData(int a, int b):data1(a),data2(b) //this is initial {} */ } 2 不同cpp文