关于c++拷贝构造函数

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了关于c++拷贝构造函数相关的知识,希望对你有一定的参考价值。

菜鸟问题,高手不要笑.
头文件:Point4.h
#include<iostream.h>
class Point

private:
double x,y;
public:
Point()

x=0.0;
y=0.0;

Point(double x1,double y1)

x=x1;
y=y1;
cout<<"构造函数被调用.\n";

Point(Point &p);

double Xcoord()

return x;

double Ycoord()

return y;

;
Point::Point(Point &p)

x=p.x;
y=p.y;
cout<<"拷贝函数被调用.\n";

源文件:
#include<iostream.h>
#include"Point4.h"
void main()

Point p1(4,5),p2(p1);
cout<<"p2=("<<p2.Xcoord()<<","<<p2.Ycoord()<<")\n";

此时能正常执行,但为什么将头文件中的拷贝函数改成:
Point(Point p);
后就提示错误?类难道不能传递给形参吗?

拷贝构造函数的参数只接受引用
因为当你直接把一个对象当参数传给函数时,实际上又要调用拷贝构造函数把它赋给形参,比如:
Point a;
Point b(a);
那么b的构造函数Point(Point p)首先要执行Point p(a),这样就进入逻辑死循环了
参考技术A 死背吧
Point(Point p);
这句话本身就是错的。语法错。

c++拷贝构造函数

1.什么是拷贝构造函数:

拷贝构造函数嘛,当然就是拷贝和构造了。(其实很多名字,只要静下心来想一想,就真的是顾名思义呀)拷贝又称复制,因此拷贝构造函数又称复制构造函数。百度百科上是这样说的:拷贝构造函数,是一种特殊的构造函数,它由编译器调用来完成一些基于同一类的其他对象的构建及初始化。其唯一的参数(对象的引用)是不可变的(const类型)。此函数经常用在函数调用时用户定义类型的值传递及返回

 

2.拷贝构造函数的形式

Class X

{

public:

  X();

  X(const X&);//拷贝构造函数

}

2.1为什么拷贝构造参数是引用类型?

其原因如下:当一个对象以传递值的方式传一个函数的时候,拷贝构造函数自动被调用来生成函数中的对象(符合拷贝构造函数调用的情况)。如果一个对象是被传入自己的拷贝构造函数,它的拷贝构造函数将会被调用来拷贝这个对象,这样复制才可以传入它自己的拷贝构造函数,这会导致无限循环直至栈溢出(Stack Overflow)。

 

3.拷贝构造函数调用的三种形式

3.1.一个对象作为函数参数,以值传递的方式传入函数体;

3.2.一个对象作为函数返回值,以值传递的方式从函数返回;

3.3.一个对象用于给另外一个对象进行初始化(常称为复制初始化)。

总结:当某对象是按值传递时(无论是作为函数参数,还是作为函数返回值),编译器都会先建立一个此对象的临时拷贝,而在建立该临时拷贝时就会调用类的拷贝构造函数

 

4.深拷贝和浅拷贝

如果在类中没有显式地声明一个拷贝构造函数,那么,编译器将会自动生成一个默认的拷贝构造函数,该构造函数完成对象之间的位拷贝。(位拷贝又称浅拷贝,后面将进行说明。)自定义拷贝构造函数是一种良好的编程风格,它可以阻止编译器形成默认的拷贝构造函数,提高源码效率。

在某些状况下,类内成员变量需要动态开辟堆内存,如果实行位拷贝,也就是把对象里的值完全复制给另一个对象,如A=B。这时,如果B中有一个成员变量指针已经申请了内存,那A中的那个成员变量也指向同一块内存。这就出现了问题:当B把内存释放了(如:析构),这时A内的指针就是野指针了,出现运行错误。事实上这就要用到深拷贝了,要自定义拷贝构造函数。

深拷贝和浅拷贝可以简单理解为:如果一个类拥有资源,当这个类的对象发生复制过程的时候,资源重新分配,这个过程就是深拷贝,反之,没有重新分配资源,就是浅拷贝。下面举个深拷贝的例子。

#include <iostream>

using namespace std;

class CA

{

public:

  CA(int b,char* cstr)

  {

    a=b;

     str=new char[b];

    strcpy(str,cstr);

  }

  CA(const CA& C)

  {

    a=C.a;

    str=new char[a]; //深拷贝

    if(str!=0)

     strcpy(str,C.str);

  }

  void Show()

  {

    cout<<str<<endl;

  }

  ~CA()

  {

    delete str;

  }

 private:

   int a;

   char *str;

};

int main()

{

  CA A(10,"Hello!");

  CA B=A;

  B.Show();

  return 0;

}

浅拷贝资源后在释放资源的时候会产生资源归属不清的情况导致程序运行出错。一定要注意类中是否存在指针成员。

 

 

5.拷贝构造函数与“=“赋值运算符

例如:

class CExample

{}

int main()

{

CExample e1 = new CExample;

CExample e2 = e1;//调用拷贝构造函数

CExample e3(e1);//调用拷贝构造函数

CExample e4;

e4 = e1;//调用=赋值运算符

}

通常的原则是:①对于凡是包含动态分配成员或包含指针成员的类都应该提供拷贝构造函数;②在提供拷贝构造函数的同时,还应该考虑重载"="赋值操作符号。

以上是关于关于c++拷贝构造函数的主要内容,如果未能解决你的问题,请参考以下文章

c++关于派生类的拷贝构造函数

关于C++ 中 thread 的拷贝构造函数

C++类的成员函数:构造析构拷贝构造运算符重载

C++类的成员函数:构造析构拷贝构造运算符重载

C++类的成员函数:构造析构拷贝构造运算符重载

C++类的成员函数:构造析构拷贝构造运算符重载