拷贝构造函数是一种特殊的构造函数,它在创建对象时,是使用同一类中之前创建的对象来初始化新创建的对象。拷贝构造函数通常用于:
-
通过使用另一个同类型的对象来初始化新创建的对象。
-
复制对象把它作为参数传递给函数。
-
复制对象,并从函数返回这个对象。
三种方法 用代码演示
#include<iostream>
using namespace std;
class Person
{
public:
Person()
{
cout<<"Person默认构造函数调用"<<endl;
}
Person(int age) //有参构造
{
cout<<"Person有参构造函数调用"<<endl;
m_Age=age;
}
Person(const Person &p){
cout<<"Person拷贝构造函数调用"<<endl;
m_Age=p.m_Age;
}
~Person()
{
cout<<"Person析构函数调用"<<endl;
}
int m_Age;
};
//拷贝构造函数的调用时机
//1.使用一个已经创建完毕的对象来初始化一个新对象 用的最多
void test01()
{
Person p1(20);//有参
Person p2(p1);//p2调p1 拷贝
cout<<"p2的年龄为: "<<p2.m_Age<<endl;//证明拷贝成功
}
//2.值传递的方式给函数参数传值
void doWork(Person p)//值传递 值传递的本质就是会拷贝一个临时副本 形参
{
p.m_Age=500;
cout<<p.m_Age<<endl; //在这修改对底下不影响
}
void test02(){
Person p=1000;
doWork(p);//实参 形参不能传递给实参 !!
cout<<p.m_Age<<endl;//还是一千
}
//3.值方式返回局部对象
Person dowork2()//值方式来返回
{
Person p1;//局部对象
cout<<" "<<(int*)&p1<<endl;
//经过cout输出代码验证可知,用值的方式返回,并不会返回47行的p1,它是拷贝一个新的对象返回给test03()
return Person(p1);//新的对象
}
void test03()
{
Person p=dowork2();
cout<<(int*)&p<<endl;
}
int main(){
// test01();
test02();
// test03();
//Person默认构造函数调用
// 0x6ffd80
//Person拷贝构造函数调用
//Person析构函数调用
//0x6ffdd0
//Person析构函数调用
system("pause");
return 0;
}
其最常用第一种 把第一种灵活使用即可
................................................................................................................................分界线
如果在类中没有定义拷贝构造函数,编译器会自行定义一个。如果类带有指针变量,并有动态内存分配,则它必须有一个拷贝构造函数。拷贝构造函数的最常见形式如下:
classname (const classname &obj) {
// 构造函数的主体
}
在这里,obj 是一个对象引用,该对象是用于初始化另一个对象的。
#include <iostream>
using namespace std;
class Line
{
public:
int getLength( void );
Line( int len ); // 简单的构造函数
Line( const Line &obj); // 拷贝构造函数
~Line(); // 析构函数
private:
int *ptr;
};
// 成员函数定义,包括构造函数
Line::Line(int len)
{
cout << "调用构造函数" << endl;
// 为指针分配内存
ptr = new int;
*ptr = len;
}
Line::Line(const Line &obj)
{
cout << "调用拷贝构造函数并为指针 ptr 分配内存" << endl;
ptr = new int;
*ptr = *obj.ptr; // 拷贝值
}
Line::~Line(void)
{
cout << "释放内存" << endl;
delete ptr;
}
int Line::getLength( void )
{
return *ptr;
}
void display(Line obj)
{
cout << "line 大小 : " << obj.getLength() <<endl;
}
// 程序的主函数
int main( )
{
Line line(10);
display(line);
return 0;
}
当上面的代码被编译和执行时,它会产生下列结果:
调用构造函数
调用拷贝构造函数并为指针 ptr 分配内存
line 大小 : 10
释放内存
释放内存
分析
-
拷贝构造函数:
- 在拷贝构造函数
Line::Line(const Line &obj)
中,同样为ptr
分配了新的动态内存,并将其内容复制为*obj.ptr
。 - 内存分配的操作发生在
ptr = new int;
这一行。这是为了存储一个整数,该整数的值是*obj.ptr
,也就是原对象中存储的值
- 在拷贝构造函数