当前位置: 首页 > news >正文

C++语法|对象的浅拷贝和深拷贝

背景:
我们手写一个顺序栈,展开接下来的实验:
⭐️ this指针指向的是类在内存中的起始位置

class SeqStack {
public:SqeStack(int size = 10) {cout << this << "SeqStack()" << endl;pstack_ = new int[size_];top_ = -1;size_ = size;	}~SeqStack() {cout << this << " ~SeqStack()" << endl;delete[] pstack_;pstack_ = nullptr;}void push(int val) {if (full()) resize();pstack_[++top_] = val;}void pop() {if (empty()) return;--top_;}int top() { return pstack_[top_]; }bool empty() { return top_ == -1; }bool full() { return top_ == size_ - 1; }
private:int *pstack_; //动态开辟数组,存储顺序栈的元素int top_; //指向栈顶元素的位置int size_; //数组扩容的总大小void resize () {int *ptem = new int[size_ * 2];for (int i = 0; i < size_; i++) {ptmp[i] = pstack_[i];}delete[] pstack_;pstack_ = ptmp;size_ *= 2}
};

文章目录

  • 浅拷贝
  • 自定义拷贝构造
  • 为什么不用memcpy
  • 赋值操作引起的浅拷贝问题
  • 重载赋值运算符

浅拷贝

int main () {SeqStack s1(10);SeqStack s2 = s1;  //#1//SeqStack s3 = s1; #2 1和2都是调用拷贝构造//SeqStack s4; s4 = s1 #这个才是调用operator=return 0;
}

程序会直接崩溃,终端提示我们两次释放相同的内存空间,造成内存泄漏。
这是由于代码SeqStack s2 = s1;明显是调用了默认的拷贝构造函数,默认的拷贝构造其实是一个浅拷贝。
可以看如下图:
在这里插入图片描述
所以说在析构的时候,先析构s2,导致我们new int[10]这块内存就已经没有了,s1中的pstack_成为了一个野指针,我们在析构一个已经释放了的内存。

总结:
对象默认的拷贝构造事做内存的数据拷贝。
关键是对象如果占用了外部资源,那么浅拷贝就出现问题了。
如果之后我们发现一个对象有一个指针,并且这个指针还指向一个外部的(堆)上的内存,所以我们一定要警防浅拷贝。

此时我们一定不能依靠编译器为我们自动生成的拷贝构造

自定义拷贝构造

SeqStack(const SeqStack &src) {//以下就是默认的浅拷贝操作//pstack_ = src.pstack_;//top_ = src.top_;//size_ = src.size_;pstack_ = new int[srtc.size_];for (int i = 0; i <= src.top_; ++i) {pstack_ = src.pstack_;}top_ = src.top_;size_ = src.size_;
}

在这里我们就对指针类型做了一个深拷贝。

为什么不用memcpy

我们在进行数据拷贝的时候,都是用的for循环,而不用memcpy,这是为什么呢?

在这里插入图片描述
如图所示,我们需要把小内存的数据全部放到大内存上来实现扩容,或者数据的迁移。

因为我们在进行数据拷贝的时候,假如我们要把这块内存上的数据拷贝到那块内存上,如果这块内存上的数据仅仅是里面放int型,但是每一个整型都不占用该整型之外的资源(堆上的资源),就是说这块内存本身就只是放了一块值而已。

那使用内存的memcpy拷贝到那块大内存中,那是没有任何问题的。

那我们假设一下,这个数组里面放的不是整型,而是对象,而且每一个对象里面都有指针,而且还指向了外部的资源,也就是说这个数组里面存的对象的浅拷贝是有问题的。

比如说我们的ptmp[i]里面放的不是整型而是对象,那么我用memcpoy的话就只是把对象本身的内存拷贝了一份,这做的都是浅拷贝的操作

浅拷贝的问题就是,让我们拷贝完的对象里面由于指针跟我们原来对象内存里面的指针指向的都是同一块资源,等我们拷贝完之后,删除小内存中的对象,会自动调用析构函数,析构函数会释放对象资源,那不就直接把我们打指针对象指向的那块堆内存也释放掉了。导致我们拷贝的那些对象的指针都成为了野指针。
在这里插入图片描述
所以说面向对象编程里面,数据的拷贝必须得用for循环来防止内存泄漏的问题

赋值操作引起的浅拷贝问题

int main () {SeqStack s1(10);SeqStack s2 = s1;  //#1//SeqStack s3 = s1; #2 1和2都是调用拷贝构造//SeqStack s4; s4 = s3; //赋值操作=》做直接的潜拷贝return 0;}
}

同理,如果我们不在类里面不重载赋值运算符,编译器会为我们调用默认的赋值操作。

我们仍然会在第二次析构的时候出现问题。

所以我们需要重载赋值运算符operator=

重载赋值运算符

void operator= (const SeqStack &src) {pstack_ = new int[srtc.size_];for (int i = 0; i <= src.top_; ++i) {pstack_ = src.pstack_;}top_ = src.top_;size_ = src.size_;
}

至此,完美解决。

http://www.lryc.cn/news/350283.html

相关文章:

  • 行为型模式
  • AI大模型日报#0515:Google I/O大会、 Ilya官宣离职、腾讯混元文生图大模型开源
  • 计算机网络-负载均衡算法
  • Excel Module: Iteration #1 EasyExcel生成下拉列表模版时传入动态参数查询下拉数据
  • 【回溯算法】【Python实现】TSP旅行售货员问题
  • Java处理xml
  • 软考中级-软件设计师 (十一)标准化和软件知识产权基础知识
  • pytest教程-46-钩子函数-pytest_sessionstart
  • Windows内核函数 - ASCII字符串和宽字符串
  • 从零开始学习MySQL 事务处理
  • 字符数组以及字符串相关的几个函数
  • AOP面向切面编程
  • C# WinForm —— 15 DateTimePicker 介绍
  • SpringBoot中六种批量更新Mysql 方式效率对比
  • 【SpringBoot】SpringBoot整合jasypt进行重要数据加密
  • 【Go语言入门学习笔记】Part1.梦开始的地方
  • 数据特征降维 | 主成分分析(PCA)附Python代码
  • 当服务实例出现故障时,Nacos如何处理?
  • 遥感数据集制作(Potsdam数据集为例):TIF图像转JPG,TIF标签转PNG,图像重叠裁剪
  • 根据web访问日志,封禁请求量异常的IP,如IP在半小 时后恢复正常则解除封禁
  • 2.go语言初始(二)
  • MQTT对比HTTP
  • 暴力数据结构之二叉树(堆的相关知识)
  • 死锁调试技巧:工作线程和用户界面线程
  • 蓝桥杯-外卖店优先级(简单写法)
  • VueRouter使用总结
  • Flink checkpoint 源码分析- Checkpoint snapshot 处理流程
  • Leaflet.canvaslabel在Ajax异步请求时bindPopup无效的解决办法
  • Go 处理错误
  • python读取excel数据写入mysql