黑马网站建设,贵阳市做网站的公司有哪些,建立门户网站多少钱,wordpress 七牛视频教程2.6 对象之间的复制一、要点归纳1. 对象之间的复制操作1.1 运算符1.2 拷贝构造函数2. 对象之间的浅复制和深复制2.1 对象的浅复制2.2 对象的深复制二、面试真题解析面试题1面试题2一、要点归纳
1. 对象之间的复制操作 同一个类的对象之间可以进行复制操作#xff0c;即将一个…
2.6 对象之间的复制一、要点归纳1. 对象之间的复制操作1.1 运算符1.2 拷贝构造函数2. 对象之间的浅复制和深复制2.1 对象的浅复制2.2 对象的深复制二、面试真题解析面试题1面试题2一、要点归纳
1. 对象之间的复制操作 同一个类的对象之间可以进行复制操作即将一个对象数据成员复制给另一个对象的相应的数据成员。
1.1 运算符 当已经定义了同一个类的多个对象时可以在这些对象之间进行赋值运算通常采用运算符来实现例如
MyClass s1,s2;
s1s2;上述对象复制语句的功能只是将s2的所有数据成员赋值给s1相应的数据成员。如果程序员在类中没有定义运算符编译器会相建立默认构造函数那样建立一个默认的运算符。实际上运算符对应operator成员函数所谓定义运算符就是重载operator成员函数。
1.2 拷贝构造函数 如果创建新对象并赋值调用拷贝构造函数。在定义新对象并初始化时形如A xy的语句是通过调用拷贝构造函数创建对象x通常将对象y的数据成员赋值给对象x相应的数据成员例如有以下程序
class A
{int m;public:A(){m 0;std::cout 默认构造函数 std::endl;}~A(){std::cout 析构函数 std::endl;}A(int n){m n;std::cout 重载构造函数 std::endl;}A(const A b){std::cout 拷贝构造函数 std::endl;}A operator(const A b){std::cout operator std::endl;return *this;}
};void test01()
{A a(2), b(a), c;c a;A d a;
}输出结果如下
重载构造函数
拷贝构造函数
默认构造函数
operator
拷贝构造函数
析构函数
析构函数
析构函数
析构函数建立A a(2)调用重载构造函数执行b(a)调用拷贝构造函数执行A c调用默认构造函数。执行ca调用operator成员函数执行A da调用拷贝构造函数。依次销毁d\c\b\a调用析构函数。
2. 对象之间的浅复制和深复制 无论是拷贝构造函数还是运算符都可春实现对象复制。对象复制又分为浅复制和深复制。
2.1 对象的浅复制 当两个对象之间进行复制时若复制完成后她们还共享某些资源内存空间其中一个对象的销毁会影响另一个对象这种对象之间的复制称为浅复制。例如有以下程序
class Student
{int no;char *pname;public:Student() {} // 默认构造函数Student(int n, char *p) // 重载构造函数{no n;pname new char[10]; // 用new分配内存空间strcpy(pname, p);}Student(Student s) // 拷贝构造函数{no s.no;pname s.pname;}void display(){std::cout no: no ,name: pname std::endl;std::cout pname: (int *)pname std::endl; // 输出pname地址}
};
void test02()
{Student s(10, (char *)Mary), t(s);std::cout s:;s.display();std::cout t:;t.display();
}输出如下
s:no:10,name:Mary
pname:0x55d1366c22c0
t:no:10,name:Mary
pname:0x55d1366c22c0在上述程序中先声明了一个类Student其中有一个拷贝构造函数。在main函数中建立了一个对象s通过拷贝构造函数由s建立t对象。由Student类的拷贝构造函数可以看到s到t的复制是浅复制如下图所示因为这两个对象的pname数据成员均指向相同的内存空间。
2.2 对象的深复制 当两个对象之间进行复制时若复制完成后她们不会共享任何资源内存空间其中一个对象的销毁不会影响到另一个对象这种对象之间的复制称为深复制。例如将上述程序修改如下
class Student
{int no;char *pname;public:Student() {} // 默认构造函数Student(int n, char *p) // 重载构造函数{no n;pname new char[10]; // 用new分配内存空间strcpy(pname, p);}Student(Student s) // 拷贝构造函数{no s.no;// pname s.pname;pname new char[strlen(s.pname 1)];strcpy(pname, s.pname);}void display(){std::cout no: no ,name: pname std::endl;std::cout pname: (int *)pname std::endl; // 输出pname地址}~Student() { delete pname; } // 析构函数
};void test03()
{Student s(10, (char *)Mary), t(s);std::cout s:;s.display();std::cout t:;t.display();
}s:no:10,name:Mary
pname:0x55acd2a8b2c0
t:no:10,name:Mary
pname:0x55acd2a8b2e0修改后类Student的亏被构造函数在复制数据成员pname时另外开辟内存空间从而当执行Student s(10, (char *)“Mary”), t(s);的时候由对象s复制产生对象t时这两队对象的pname成员分别指向不同的内存单元如图
二、面试真题解析
面试题1
【面试题】什么时候必须重写拷贝构造函数 【答】当构造函数涉及动态存储分配空间时要自己写拷贝构造函数并且要深复制。
面试题2
【面试题】不考虑任何编译器优化下面代码的第8行会发生一次拷贝构造、一次析构函数、一次operator
#include iostream
class B
{
};
B func(const B rhs)
{return rhs;
}
void main()
{B b1,b2;b2func(b1);
}【答】首先调用一次拷贝构造函数将b1复制给形参rhs返回rhs并赋值给b2调用一次operatorfunc函数执行完毕销毁rhs调用一次析构函数。