拷贝构造函数

目录

1.拷贝构造函数

2.拷贝构造函数功能--用已有对象去构造本类的新对象(倒数两行)

3.如果程序员没有写,类会生成默认的拷贝构造函数,功能是用已有对象的值依次给新对象的值初始化

4.--拷贝构造函数调用的3种情况

5.默认拷贝构造--浅拷贝--值之间的拷贝

6.深拷贝

7.类会默认提供的函数--6个

1.拷贝构造函数

拷贝构造函数--本类旧对象去构造本类新对象 1--参数的类型 值类型(不行,会产生递归),指针类型(会出现歧义),引用类型(本类对象的const引用)

class A
{
public:
	A(int i = 0,int j = 0):m_i(i),m_j(j){cout<<"A"<<endl;}
	void Print()
	{
		cout<<"m_i = "<<m_i<<"  m_j = "<<m_j<<endl;
	}
 private:
    int m_i;
    iny m_j;
};
void main()
{
    A a(3,6);
    a.Print;//正常打印出m_i是3,m_j是6
    A b=a;//声明一个对象,想用对象a的值给b赋值
          //本质其实就是用a的数据成员去初始化b 的数据成员
    b.Print();
}

截图展示:

结论:由上图可知:b是一个对象,用a去初始化是成功的,但b的产生会调用构造函数,所以b调用了一个特别的构造函数.

1)值类型(不行,会产生递归) 

class A
{
public:
	A(int i = 0,int j = 0):m_i(i),m_j(j){cout<<"A"<<endl;}
	void Print()
	{
		cout<<"m_i = "<<m_i<<"  m_j = "<<m_j<<endl;
	}
	A(A s)//假设调用的是这个函数,第一步用a去给s初始化,那a是个对象,又要调这个函数,就构成递归了
	{
	
	}
 private:
    int m_i;
    iny m_j;
};
void main()
{
    A a(3,6);
    a.Print;//正常打印出m_i是3,m_j是6
    A b=a;//声明一个对象,想用对象a的值给b赋值
          //本质其实就是用a的数据成员去初始化b 的数据成员
    b.Print();
}

 2)指针类型

class A
{
public:
	A(int i = 0,int j = 0):m_i(i),m_j(j){cout<<"A"<<endl;}
	void Print()
	{
		cout<<"m_i = "<<m_i<<"  m_j = "<<m_j<<endl;
	}
    A (A *s):m_i(s->m_i),m_j(s->m_j)//假设调这个函数,
    {
        cout<<"A(A*)"<<endl;
    }

 private:
    int m_i;
    iny m_j;
};
void main()
{
    A a(3,6);
    a.Print;//正常打印出m_i是3,m_j是6
    A b=&a;//可能会理解为用a的地址给b赋值
    b.Print();
}

截图展示:  

由上图可知:用指针功能可以实现,但是会出现歧义, A b=&a;//可能会理解为用a的地址给b赋值

3)引用类型(本类对象的const引用)

class A
{
public:
	A(int i = 0,int j = 0):m_i(i),m_j(j){cout<<"A"<<endl;}
	void Print()
	{
		cout<<"m_i = "<<m_i<<"  m_j = "<<m_j<<endl;
	}
    A(const A &s):m_i(s.m_i),m_j(s.m_j)//s是a的别名
    {
      cout<<"A (A&)"<<endl;
    }
 private:
    int m_i;
    iny m_j;
};
void main()
{
    A a(3,6);
    a.Print;//正常打印出m_i是3,m_j是6
    A b=a;//声明一个对象,想用对象a的值给b赋值
          //本质其实就是用a的数据成员去初始化b 的数据成员
    b.Print();
}

截图展示: 

A(const A &s)就是拷贝(复制)构造函数。

2.拷贝构造函数功能--用已有对象去构造本类的新对象(倒数两行)

class A
{
public:
	A(int i = 0,int j = 0):m_i(i),m_j(j){cout<<"A"<<endl;}
	void Print()
	{
		cout<<"m_i = "<<m_i<<"  m_j = "<<m_j<<endl;
	}
    A(const A &s):m_i(s.m_i),m_j(s.m_j)
    {
      cout<<"A (A&)"<<endl;
    }
 private:
    int m_i;
    iny m_j;
};
void main()
{
    A a(3,6);
    a.Print;//正常打印出m_i是3,m_j是6
    A b=a;//声明一个对象,想用对象a的值给b赋值
          //本质其实就是用a的数据成员去初始化b 的数据成员
    b.Print();
    A c;//已经构造了一个c
    c=a;//在这里c是旧对象了,这句话是用a给c重新赋值
         //不会调用拷贝构造函数,c在上面A c已经构造出来了,调用的是后面要学的赋值运算符重载
}

 

3.如果程序员没有写,类会生成默认的拷贝构造函数,功能是用已有对象的值依次给新对象的值初始化

4.--拷贝构造函数调用的3种情况

1--用已有对象去初始化本类的对象

如上述所示:A b=a;

2--函数传参,类类型的值传递

class A
{
public:
	A(int i = 0,int j = 0):m_i(i),m_j(j){cout<<"A"<<endl;}
	void Print()
	{
		cout<<"m_i = "<<m_i<<"  m_j = "<<m_j<<endl;
	}


 private:
    int m_i;
    iny m_j;
};
void Test(A t)//A t=c从实参c到形参t的过程,调用拷贝构造

{

   cout<<"Test:"<<endl;

   t.Print();

}

void main()
{
    A a(2,6);//调用构造函数A(int,int)
    A b(a);//调拷贝构造函数A(A&)
    A c(9,3);//调用构造函数A(int,int)
    Test(c);//c是旧对象 t是新对象
}

截图展示:  

 

  执行过程:

问:输出多少句话 (10句话)

 

解释:

1:调用A构造函数构造对象a,输出A 2:拷贝构造函数用a构造出b ,输出A(A&)

3:调用构造函数构造c ,输出A

4:c是旧对象,t是新对象,所以先拷贝构造出t,输出A(A&)

5:调用Test函数,输出Test;

6:t.Print()调用Print函数,输出m_i和m_j的值

7:程序即将结束,根据先构造后析构的原则,先析构t,输出~A

8:析构c,输出~A

9:析构b ,输出~A

10:析构a ,输出~A

3--函数是类类型的值返回 局部对象->临时对象

代码展示;

class A
{
public:
	A(int i = 0,int j = 0):m_i(i),m_j(j){cout<<"A"<<endl;}
	void Print()
	{
		cout<<"m_i = "<<m_i<<"  m_j = "<<m_j<<endl;
	}
	A(const A &s):m_i(s.m_i),m_j(s.m_j)
	{
	cout<<"A(A&)"<<endl;
	}
    ~A()
    {
        cout<<"~A"<<endl;
    }
 private:
    int m_i;
    iny m_j;
};
void Test(A t)//A t=c从实参c到形参t的过程,调用拷贝构造

{

   cout<<"Test:"<<endl;

   t.Print();

}

void main()
{
    A a(2,6);//调用构造函数A(int,int)
    A b(a);//调拷贝构造函数A(A&)
    A c(9,3);//调用构造函数A(int,int)
    Test(c);//c是旧对象 t是新对象
}

 截图展示:

3)函数是类类型的值返回,由局部对象到临时对象调用拷贝构造函数

class A
{
public:
	A(int i = 0,int j = 0):m_i(i),m_j(j){cout<<"A"<<endl;}
	void Print()
	{
		cout<<"m_i = "<<m_i<<"  m_j = "<<m_j<<endl;
	}
	A(const A &s):m_i(s.m_i),m_j(s.m_j)
	{
	cout<<"A(A&)"<<endl;
	}
    ~A()
    {
        cout<<"~A"<<m_i<<m_j<<endl;
    }
 private:
    int m_i;
    iny m_j;
};
void Test(A t)//A t=c从实参c到形参t的过程,调用拷贝构造

{

   cout<<"Test:"<<endl;

   t.Print();

}
A fn()
{
    A d(10,12);
    return d;
}

void main()
{
    A a(2,6);//调用构造函数A(int,int)
    A b(a);//调拷贝构造函数A(A&)
    A c(9,3);//调用构造函数A(int,int)
    Test(c);//c是旧对象 t是新对象
    c=fn();
    cout<<"main end"<<endl;
}

 分析过程:

main和fn是两个栈区,他们之间不能直接的传递数据,所以在生成局部对象d的时候,其中也有一个临时对象temp去接受d的值,随后把值传给main函数中的c。

结果输出:  

A:构造A对象,调用普通构造函数

A(A &):调用拷贝构造构造对象b

A:调用普通构造构造c

A(A &):进入Test函数,c是旧对象,t是新对象,所以先拷贝构造出t

Test:Test函数自身打印Test

m_i=9 m_j=3:t.Print函数调Print,打印m_i和m_j的值

A~93:Test函数即将结束,析构t对象

A:进入fn函数,调用普通构造构造对象d

A(A &):因为栈区原因,d无法带回值回到main函数,所有有一个临时对象,调用拷贝构造函数构造临时对象

A~1012:析构局部对象d

A~1012:临时对象把值带给c之后,析构临时对象

main end:主函数打印

A~1012:析构c

A~26:析构b

A~26:析构a

5.默认拷贝构造--浅拷贝--值之间的拷贝

class Student
{
public:
	Student(int num=0,char *name = "oo",char sex='m'):m_num(num),m_sex(sex)
	{
		m_name = new char[strlen(name)+1];
		strcpy(m_name,name);
    }
	~Student()
	{
		delete[]m_name;
	}
	void Print()
	{
		cout<<m_num<<" "<<m_name<<" "<<m_sex<<endl;
	}
private:
	int m_num;
	char *m_name;
	char m_sex;
};
void main()
{
	Student s(1001,"lisi",'f');
	s.Print();
	Student s1(s); //调用默认拷贝构造 ,导致s1对象中的m_name和s对象中的m_name指向的内存相同
	s1.Print();
	cout<<sizeof(s)<<endl;
}

截图结果:  

 

结论:功能实现了,但是出现了崩溃,原因是浅拷贝在遇到指针的时候,原有的对象s拷贝构造了一个s1,s和s1里的函数指针共同指向同一段字符串的首地址,在析构这两对象时,s1被析构后,指针所有的内存单元被释放,等到析构s1的时候,这一段已经空的内存单元又被析构,就会崩溃。

由上图可知:b是一个对象,用a去初始化是成功的,但b的产生会调用构造函数,所以b调用了一个特别的构造函数

6.深拷贝

深拷贝-- 如果有指针作为数据成员,而且用本类的旧对象去构造了新对象,则要写拷贝构造 给新对象的指针要去开辟和旧对象指针同样大小的内存单元,然后拷贝相同的值 1.不但拷贝值,而且要拷贝资源

class Student
{
public:
	Student(int num=0,char *name = "oo",char sex='m'):m_num(num),m_sex(sex)
	{
		m_name = new char[strlen(name)+1];
		strcpy(m_name,name);
	}
	Student(const Student&s):m_num(s.m_num),m_sex(s.m_sex) //深拷贝
	{
		m_name = new char[strlen(s.m_name)+1];//开辟和旧对象的字符串一样大小的内存
		strcpy(m_name,s.m_name);
	}
	~Student()
	{
		delete[]m_name;
	}
	void Print()
	{
		cout<<m_num<<" "<<m_name<<" "<<m_sex<<endl;
	}
private:
	int m_num;
	char *m_name;
	char m_sex;
};
void main()
{
	Student s(1001,"lisi",'f');
	s.Print();
	Student s1(s); //调用默认拷贝构造 ,导致s1对象中的m_name和s对象中的m_name指向的内存相同
	s1.Print();
	cout<<sizeof(s)<<endl;
}
类会提供默认的函数--6个
1.构造函数
2.析构函数
3.拷贝构造(复制构造)
  --只要有指针作为数据成员
4.运算符重载赋值=
*/
class Student
{
public:
	Student(int num=0,char *name = "oo",char sex='m'):m_num(num),m_sex(sex)
	{
		m_name = new char[strlen(name)+1];
		strcpy(m_name,name);
	}
	Student(const Student&s):m_num(s.m_num),m_sex(s.m_sex) //深拷贝
	{
		m_name = new char[strlen(s.m_name)+1];//开辟和旧对象的字符串一样大小的内存
		strcpy(m_name,s.m_name);
	}
	~Student()
	{
		delete[]m_name;
	}
	void Print()
	{
		cout<<m_num<<" "<<m_name<<" "<<m_sex<<endl;
	}
private:
	int m_num;
	char *m_name;
	char m_sex;
};
void main()
{
	Student s(1001,"lisi",'f');
	s.Print();
	Student s1(s); //调用默认拷贝构造 ,导致s1对象中的m_name和s对象中的m_name指向的内存相同
	s1.Print();
	cout<<sizeof(s)<<endl;
}

 结果展示:

7.类会默认提供的函数--6个

1.构造函数

2.析构函数

3.拷贝构造(复制拷贝)--只要有指针作为数据成员

4.运算符重载=

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

我在凌晨等太阳¤

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值