定义:
拷贝构造函数是C++中的一种特殊的构造函数,用于创建一个对象的副本。它的作用是在对象之间进行拷贝初始化。拷贝构造函数通常以引用的方式接受同类型的对象作为参数,以完成对象的拷贝。程序中如果没有定义拷贝构造函数系统会生成一个默认的拷贝构造函数,它会将作为初始值的对象的数据成员的值都拷贝到要初始化的对象中。
注意:拷贝构造函数的形参通常建议使用引用类型,如果不使用引用类型,而是传递对象本身,会导致在函数调用时发生对象的拷贝,这样就会调用拷贝构造函数本身,导致无限循环。因此,为了避免这种情况,拷贝构造函数的形参一般建议使用引用类型
定义拷贝构造函数的形式是:
class 类名
{
public :
类名(形参); //构造函数
类名(类名 &对象名); //拷贝构造函数
...
};
类名::类(类名 &对象名) //拷贝构造函数的实现
{
函数体
}
使用场景:
拷贝构造函数在以下三种情况下会被调用:
1.当用类的一个对象去初始化该类的另一个对象时系统自动调用拷贝构造函数实现拷贝赋值。
int main()
{
Point A(1,2);
Point B(A); //拷贝构造函数被调用
cout<<B.GetX()<<endl;
return 0;
}
2.若函数的形参为类对象,调用函数时,实参赋值给形参,系统自动调用拷贝构造函数。例如:
void fun1(Point p)
{
cout<<p.GetX()<<endl;
}
int main()
{
Point A(1,2);
fun1(A); //调用拷贝构造函数
return 0;
}
3.当函数的返回值是类对象时,系统自动调用拷贝构造函数。例如:
Point fun2()
{
Point A(1,2);
return A; //调用拷贝构造函数
}
int main()
{
Point B;
B=fun2();
return 0;
}
最后这种情况怎么调用的拷贝构造函数呢?对象A 是局部对象,在fun2 函数执行完就释放了,那怎么将它拷贝给对象B
呢?编译器在执行B=fun2()时会创建一个临时的无名对象,在执行return A 时实际上是调用了拷贝构造函数将A 的值拷贝到
了临时对象中,A 就释放了,然后将临时对象的值再拷贝到对象B 中。
延伸:
在C#中,没有类似于C++的拷贝构造函数机制。C#中的对象赋值和参数传递都是通过引用进行的,而不是像C++中那样通过值传递。
我们来看一个例子来演示C#和C++在参数传递时是否会对传入实参产生影响:
C#代码:
using System;public class Example
{public int X { get; set; }
}public class Program
{public static void ModifyObject(Example obj){obj.X = 10; // 修改形参}public static void Main(string[] args){Example original = new Example();original.X = 5;ModifyObject(original); // 传递对象给函数Console.WriteLine(original.X); // 输出:10}
}
C++代码:
#include <iostream>class Example {
public:int X;
};void ModifyObject(Example obj)
{obj.X = 10; // 修改形参
}int main()
{Example original;original.X = 5;ModifyObject(original); // 传递对象给函数std::cout << original.X << std::endl; // 输出:5return 0;
}
可以看出C#程序在对象 original
被传递给 ModifyObject
函数,虽然在函数内部修改了形参 obj
的属性值,但是原始对象 original
的属性值也被修改了。这是因为 obj
和 original
实际上引用的是同一个对象。而C++程序虽然在 ModifyObject
函数中修改了形参 obj
的属性值,但是原始对象 original
的属性值没有受到影响。这是因为在调用 ModifyObject
函数时,发生了对象的复制,形参 obj
是原始对象的一个副本,对形参的修改不会影响原始对象。