【C++程序员的自我修炼】拷贝构造函数

这篇具有很好参考价值的文章主要介绍了【C++程序员的自我修炼】拷贝构造函数。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

心存希冀

追光而遇目有繁星

沐光而行

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法


目录

拷贝构造函数概念

拷贝构造的特征

无穷递归的解释

浅拷贝

总结:

 深拷贝

拷贝构造函数典型调用场景

总结 

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

契子

在生活中总有很多琐事,不做不行做了又怕麻烦,有时候想要是有个和自己一模一样的人就好了

可以帮我上早读晚修~

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

就像以上的两个安妮娅一样,可以一个上学一个宅在家看电视
那在创建对象时,可否创建一个与已存在对象一某一样的新对象呢?
答曰 ~ 当然可以,你在现实中办不到了事情,C++都可以帮你做到,不管是对象还是 -- 分身💞

拷贝构造函数概念

拷贝构造函数只有单个形参 ,该形参是对本 类类型对象的引用 ( 一般常用 const 修饰 ) ,在用 已存 在的类类型对象创建新对象时由编译器自动调用
简单来说就是: 使用同类型的对象拷贝初始化

为了能让各位老铁更清楚的认识 拷贝构造函数 的概念,我们先小小的举个栗子~

#include<iostream>
using std::cout;
using std::endl;

class Date
{
public:
	Date(int year = 2024, int month = 4, int day = 13)
	{
		_year = year;
		_month = month;
		_day = day;
	}
	Date(const Date& d)
	{
		_year = d._year;
		_month = d._month;
		_day = d._day;
	}
	void Print()
	{
		cout << " year = " << _year << " month = " << _month << " day = " << _day << endl;
	}
private:
	int _year;
	int _month;
	int _day;
};

int main()
{
	Date d1(2024, 4, 14);
	d1.Print();
	Date d2(d1);
	d2.Print();
	return 0;
}

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

Date d2(d1);

以上是拷贝构造的一种写法,以下提供另一种写法跟 赋值 很像

Date d2 = d1;

我们发现此时的 d2 已经具备了 d1 的所有特征

拷贝构造的特征

我们来总结一下拷贝构造的特点:

拷贝构造函数 是构造函数的一个重载形式
拷贝构造函数的 参数只有一个 必须是类类型对象的引用 ,使用 传值方式编译器直接报错
因为会引发无穷递归调用

为什么会引发 无穷递归 呢?

我们来看

无穷递归的解释

#include<iostream>
using std::cout;
using std::endl;

class Date
{
public:
	Date(int year = 2024, int month = 4, int day = 13)
	{
		_year = year;
		_month = month;
		_day = day;
	}
	Date(const Date& d)
	{
		cout << "Date(const Date & d)" << endl;
		_year = d._year;
		_month = d._month;
		_day = d._day;
	}
	void Print()
	{
		cout << "year = " << _year << " month = " << _month << " day = " << _day << endl;
	}
private:
	int _year;
	int _month;
	int _day;
};

void Fun(Date d)
{
	d.Print();
}

int main()
{
	Date d1(2024, 4, 15);
	Fun(d1);
	return 0;
}

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

我们发现在上面这段程序中竟然发生了 拷贝构造 

调用 Fun 得先传参而拷贝构造也是在传参中触发的

总结:

自定义类型对象传值传参要调用拷贝构造来完成

那么有没有什么方法有不调用拷贝构造呢 ?还真有而且有两种

传指针

void Fun(Date* d)
{
	d->Print();
}
int main()
{
	Date d1(2024, 4, 15);
	Fun(&d1);
	return 0;
}

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

为什么传指针就可以呢? 

因为此时传的是 d1 的地址,内置类型相当于我把地址拷贝给你

但是可以归可以,但是这样做了的话,就已经不叫拷贝构造函数了,而就是一个以指针变量为形参的构造函数

传引用

void Fun(Date& d)
{
	d.Print();
}
int main()
{
	Date d1(2024, 4, 15);
	Fun(d1);
	return 0;
}

这里相当于给 d1 取了别名 d ,实际还是那块地址空间

我们画个图来理解一下:
【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

 步骤:

<1>当一个对象以值方式传递时,编译器会生成代码调用它的拷贝构造函数生成一个复本

<2>当我们以传值的方式传递时,我们给出一个类实例的实参,然后我们都知道我们会得到一个一样的形参
<3>创建这个形参时,编译器会自动调用类的拷贝构造函数来完成,于是在调用创建形参的拷贝函数时,又需要再次创建另外的形参,于是就一直循环下去

浅拷贝

若未显式定义,编译器会生成默认的拷贝构造函数。 默认的拷贝构造函数对象按内存存储按字节序完成拷贝,这种拷贝叫做浅拷贝,或者值拷贝
简单来说就是:不写拷贝构造编译器自动提供的就叫浅拷贝
#include<iostream>
using std::cout;
using std::endl;

class Date
{
public:
	Date(int year = 2024, int month = 4, int day = 13)
	{
		_year = year;
		_month = month;
		_day = day;
	}
	void Print()
	{
		cout << "year = " << _year << " month = " << _month << " day = " << _day << endl;
	}
private:
	int _year;
	int _month;
	int _day;
};

int main()
{
	Date d1(2024, 4, 15);
	Date d2(d1);
	d2.Print();
	return 0;
}

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

这个不写拷贝构造函数依然可以实现我们的拷贝构造~

注意:

在编译器生成的默认拷贝构造函数中,内置类型是按照字节方式直接拷贝的,而自定义类型是调用其拷贝构造函数完成拷贝的

既然那么爽不写也能用,那么我还写拷贝构造干嘛呢???

我们来看(这其实有局限性)~ 举个例子 -- 栈

错误示范

#include<iostream>
#include<assert.h>
#include<cstdlib>
using std::cout;
using std::endl;
using StackDataUsing = int;
class Stack
{
public:
	Stack(int n);
	~Stack();
	void push(StackDataUsing x);
	StackDataUsing Top();
	void Pop();
	bool Empty();
private:
	StackDataUsing* data;
	int top;
	int capacity;
};

Stack::Stack(int n = 4)
{
	StackDataUsing* newNode = new StackDataUsing[n];
	if (newNode == nullptr)
	{
		perror("new1");
		exit(-1);
	}
	data = newNode;
	capacity = n;
	top = 0;
}

Stack::~Stack()
{
	free(data);
	data = nullptr;
	top = capacity = 0;
}

void Stack::push(StackDataUsing x)
{
	if (capacity == top)
	{
		StackDataUsing* newNode = new StackDataUsing[2 * capacity];
		if (newNode == NULL)
		{
			perror("new2");
			exit(-1);
		}
		data = newNode;
		capacity *= 2;
	}
	data[top++] = x;
}

StackDataUsing Stack::Top()
{
	return data[top - 1];
}
void Stack::Pop()
{
	assert(top > 0);
	top--;
}
bool Stack::Empty()
{
	return top == 0;
}

int main()
{
	Stack st1;
	st1.push(1);
	st1.push(2);
	st1.push(3);
	Stack st2(st1);
	return 0;
}

我们想拷贝栈中 st1 的元素能成功吗

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

我们从监视的角度看,好像看起来已经成功了 st1 所有的特性都对的上

但是~

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

不知道各位老铁有没有发现他们的空间是完全一样

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法 🌤️报了个完美的错误!!!

浅拷贝,也就是值拷贝是一个字节一个字节的拷贝,但是涉及到资源申请时,还需要慎重考虑

举个栗子:

C语言中结构体传参都是传地址,一旦是传值传参,就是浅拷贝,有可能会出现 bug ,因为如果传了 malloc 开辟的地址,结构体传过来修改了,就可能改了原来的结构体。这是C语言的 bug

栈也是这样~

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

我们的 top、capacity 这样拷贝没问题,但是问题出在了 data 上,因为这里是指针也就是说将 st1data 所指向的空间地址拷贝给了 st2 ,它们两的 data 指向同一块空间,程序结束后自然要析构的,这样就导致对同一块空间析构了两次,致使程序报错

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

总结:

类中如果没有涉及资源申请时,拷贝构造函数是否写都可以;一旦涉及到资源申请时,则拷贝构造函数是一定要写

 深拷贝

那么以上栈的代码该怎么拷贝构造呢?

C++的命名风格很独特~有浅拷贝自然就有深拷贝,涉及资源申请的拷贝都要用深拷贝

先做个小总结:

浅拷贝 只是复制了对象的引用地址,两个对象指向同一个内存地址,所以修改其中任意的值,另一个值都会随之变化
深拷贝 开辟和原来一样大的空间,并将对象及值复制过来,两个对象修改其中任意的值另一个值不会改变
栈的深拷贝写法:
Stack(const  Stack& st)
{
	data = (StackDataUsing*)malloc(sizeof(StackDataUsing) * st.capacity);
	if (data == nullptr)
	{
		perror("malloc");
		return;
	}
	memcpy(data, st.data, sizeof(StackDataUsing) * st.top);
	top = st.top;
	capacity = st.capacity;
}

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

这样就没有任何报错了~我们来看是两个不同的空间哎!!!

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

这也就证明了深拷贝就是:开辟和原来一样大的空间,并将对象及值复制过来,两个对象修改其中任意的值另一个值不会改变

拷贝构造函数典型调用场景

使用已存在对象创建新对象
函数参数类型为类类型对象
函数返回值类型为类类型对象
class Date
{
public:
 Date(int year, int minute, int day)
 {
 cout << "Date(int,int,int):" << this << endl;
 }
 Date(const Date& d)
 {
 cout << "Date(const Date& d):" << this << endl;
 }
 ~Date()
 {
 cout << "~Date():" << this << endl;
 }
private:
 int _year;
 int _month;
 int _day;
};
Date Test(Date d)
{
 Date temp(d);
 return temp;
}
int main()
{
 Date d1(2022,1,13);
 Test(d1);
 return 0;
}

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

为了提高程序效率,一般对象传参时,尽量使用引用类型,返回时根据实际场景,能用引用尽量使用引用

总结 

如果没有管理资源,一般情况不需要写拷贝构造,默认生成的拷贝构造就可以
如果都是自定义类型成员,内置类型成员没有指向资源,也类似默认生成的拷贝构造就可以
一般情况下,不需要显示写析构函数,就不需要写拷贝构造
如果内部有指针、一些值指向资源,需要显示写析构释放,通常就需要显示写构造完成深拷贝

【C++程序员的自我修炼】拷贝构造函数,C++,c++,开发语言,算法

先介绍到这里啦~

有不对的地方请指出💞文章来源地址https://www.toymoban.com/news/detail-850752.html

到了这里,关于【C++程序员的自我修炼】拷贝构造函数的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包赞助服务器费用

相关文章

  • 跟敬姐读《程序员的自我修养》

    最近刚读完陈逸鹤老师的《程序员的自我修养》,结合敬姐十几年的程序员生涯,分享给大家。 敬姐也曾经有过这种想法,读过的书,不管当时觉得多么震撼和共鸣,过一段时间好像都会忘记。那我们还为什么要花费时间去阅读呢?我赞同这本书里面的一段话: 阅读就像是

    2024年02月05日
    浏览(8)
  • 程序员自我修养阅读笔记——Widnows下的动态链接

    程序员自我修养阅读笔记——Widnows下的动态链接

      Linux下的共享库为so文件为ELF格式,Windows下的共享库为dll文件PE格式。   windows下采用dll共享对象让程序更加模块化方便升级,大多数情况为 .dll 文件,也可以是``.ocx,.CPL```文件。   早期的windows中的进程并没有独立的地址空间,32bit的windows开始进程才有独立的地址空间

    2024年02月04日
    浏览(10)
  • 〖程序员的自我修养 - 认知剖析篇⑤〗- 选择前端还是后端?

    人之所以会觉得迷茫,本质上是欠缺对自己的一个控制力、识别庞杂信息、去伪存真的独立思考与认知能力。 说明:该文属于 程序员的自我修养 专栏, 购买任意白宝书体系化专栏可加入 易编程社区, 早鸟价订阅模式除外 。 福利:加入社区的小伙伴们,除了可以获取博主

    2024年02月14日
    浏览(10)
  • 〖程序员的自我修养 - 认知剖析篇②〗- 学习编程之前你需要知道这些

    人之所以会觉得迷茫,本质上是欠缺对自己的一个控制力、识别庞杂信息、去伪存真的独立思考与认知能力。 说明:该文属于 程序员的自我修养 专栏, 购买任意白宝书体系化专栏可加入 易编程社区, 早鸟价订阅模式除外 。 福利:加入社区的小伙伴们,除了可以获取博主

    2024年02月12日
    浏览(9)
  • C++:构造方法(函数);拷贝(复制)构造函数:浅拷贝、深拷贝;析构函数。

    C++:构造方法(函数);拷贝(复制)构造函数:浅拷贝、深拷贝;析构函数。

    1.构造方法(函数) 构造方法是一种特殊的成员方法,与其他成员方法不同: 构造方法的名字必须与类名相同; 无类型、可有参数、可重载 会自动生成,可自定义 一般形式:类名(形参); 例: Stu(int age); 当用户没自定义构造方法时,系统将会执行默认构造方法,当用户自定义一个

    2024年02月11日
    浏览(11)
  • 〖程序员的自我修养 - 精炼面试篇⑨〗- 技术面试不用怕,也是有套路可利用的

    人之所以会觉得迷茫,本质上是欠缺对自己的一个控制力、识别庞杂信息、去伪存真的独立思考与认知能力。 说明:该文属于 程序员的自我修养 专栏, 购买任意白宝书体系化专栏可加入 易编程社区, 早鸟价订阅模式除外 。 福利:加入社区的小伙伴们,除了可以获取博主

    2024年02月15日
    浏览(10)
  • 〖程序员的自我修养 - 认知剖析篇④〗- 关于前端方向与后端方向的一些个人见解

    人之所以会觉得迷茫,本质上是欠缺对自己的一个控制力、识别庞杂信息、去伪存真的独立思考与认知能力。 说明:该文属于 程序员的自我修养 专栏, 购买任意白宝书体系化专栏可加入 易编程社区, 早鸟价订阅模式除外 。 福利:加入社区的小伙伴们,除了可以获取博主

    2024年02月14日
    浏览(10)
  • C++ 类和对象(二)构造函数、析构函数、拷贝构造函数

    C++ 类和对象(二)构造函数、析构函数、拷贝构造函数

            本文将介绍类的6个默认成员函数中的构造函数、析构函数和拷贝构造函数,赋值重载和取地址重载涉及运算符重载的知识,将在下篇讲解。所谓默认成员函数,也就是每个类都有的成员函数,我们可以显式定义这些函数,否则,编译器会自动生成它们。 目录 前言

    2024年02月09日
    浏览(27)
  • 【C++初阶】类和对象——构造函数&&析构函数&&拷贝构造函数

    【C++初阶】类和对象——构造函数&&析构函数&&拷贝构造函数

    ========================================================================= 个人主页点击直达: 小白不是程序媛 C++系列专栏: C++头疼记 ========================================================================= 目录 前言 类的6个默认成员函数 构造函数 概念 构造函数的特性 析构函数 概念 析构函数特性 拷贝构

    2024年02月06日
    浏览(27)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包