C++复制控制:赋值操作符和析构函数

一、赋值操作符

类定义了该类型对象赋值时会发生什么。与拷贝构造函数一样,如果类没有定义自己的赋值操作符,编译器会合成一个。

1、重载操作符的简单介绍

重载操作符是一些函数,其名字为operator后跟着所定义的操作符的符号,通过定义名为operator=的函数,我们可以对赋值进行定义。操作符函数的形参表必须具有与该操作数数目相同的形参(如果操作符是一个成员,则包括隐式this形参)。赋值是二元操作符,对应的两个形参,第一个形参为左操作数,第二个形参为右操作数。

注意:

(1)当操作符为成员函数时,它的第一个操作数隐式绑定到this指针。

(2)有些操作符,例如赋值操作符必须定义为成员函数,因此赋值操作符可接受单个形参。

(3)赋值操作符返回对右操作数的引用。

 

2、合成赋值操作符

合成赋值操作符会执行逐个成员赋值:右操作数对象的每个成员赋值为左操作数对象的对应成员。对于数组,给每个数组元素赋值。

Sales_item& Sales_item::Sales_item(const Sales_item &rhs)
{
    isbn = rhs.isbn;
    units_sold = rhs.units_sold;
    revenue = rhs.revenue;
    return *this;                //返回对左操作数的引用
}

 

3、拷贝和赋值常一起使用

可以使用拷贝构造函数的类通常也可使用合成赋值操作符。一般而言,如果类需要拷贝构造函数,它也会需要赋值操作符。应将两者看作一个单元,如果需要其中一个,我们几乎也肯定需要另一个。

 

二、析构函数

析构函数的作用是完成所需资源的回收,作为类构造函数的补充。

1、何时调用析构函数

撤销类对象时自动调用析构函数:

(1)变量在超出作用域时自动撤销。例如:变量item遇到右}时。

(2)动态分配的对象只有在指向该对象的指针被删除时才撤销。例如:指针p。

Sales_item *p = new Sales_item;
{
    Sales_item item(*p);
    delete p;
}

注意:当对象的引用或指针超出作用域时,不会运行析构函数。只有删除指向动态分配对象的指针或实际对象(而不是对象的引用)超出作用域时,才会运行析构函数。

(3)撤销一个容器(不管是标准库还是内置数组)也会运行容器中元素的析构函数。

容器中的元素总是按逆序撤销,首先撤销下标为size()-1的元素,最后撤销下标为0的元素。

{
    Sales_item *p = new Sales_item[10];
    vector<Sales_item> vec(p, p + 10);
    delete [] p;
}

 

2、何时编写显式析构函数

许多类不需要显式析构函数,具有构造函数的类不一定需要定义自己的析构函数,仅在有些工作需要析构函数完成时,才需要析构函数。析构函数通常用于释放在构造函数或在对象生命期内获取的资源。

注意:

(1)如果类需要析构函数,则它也需要赋值操作符和拷贝构造函数,这是一个有用的经验法则。

(2)析构函数并不仅限于用来释放资源。一般而言,析构函数可以执行任意操作,该操作是类设计者希望在该类对象的使用完毕之后执行的。

 

3、合成析构函数

与拷贝构造函数和赋值操作符不同,编译器总会为我们合成一个析构函数。合成析构函数按对象创建时的逆序撤销每个非static成员,按成员在类中声明的逆序撤销成员。对于类类型的每个成员,合成析构函数调用该成员的析构函数撤销对象。

注意:撤销内置类型成员或复合类型成员没什么影响,合成析构函数并不删除指针成员做指向的对象。

 

4、编写析构函数

析构函数的名字前加~,没有返回值,没有形参(所以不能重载析构函数)。

注意:

(1)类可以定义多个构造函数,但只能提供一个析构函数,应用于类的所有对象。

(2)析构函数区别于拷贝构造函数和赋值操作符,即使定义了自己的析构函数,合成析构函数仍然运行。

class Sales_item
{
public:
    ~Sales_item(){}
private:
    string isbn;
    int units_sold;
    double revenue;
};

撤销Sales_item对象时,首先运行这个什么都不做的析构函数,然后再运行合成析构函数撤销类的成员。

posted @ 2015-02-21 22:15  QingLiXueShi  阅读(1126)  评论(0编辑  收藏  举报