c++中深浅拷贝以及写时拷贝的实现示例代码

发布时间 - 2026-01-11 02:35:30    点击率:

本文主要给大家介绍了关于c++中深浅拷贝及写时拷贝实现的相关内容,分享出来供大家参考学习,下面话不多说,来一起看看详细的介绍:

一:浅拷贝&深拷贝

浅拷贝:在拷贝构造的时候,直接将原内容的地址交给要拷贝的类,两个类共同指向一片空间。但是存在很大的缺陷:①一旦对s2进行操作,s1的内容也会改变;②析构时先析构s2,再析构s1,但是由于s1,s2指向同一片空间,会导致一片空间的二次析构导致出错。

深拷贝:通过开辟和源空间大小相同的空间并将内容拷贝下来再进行操作。不论是否对s2进行操作,都会拷贝一片相同大小的空间以及内容下来。

图示如下:

深拷贝实现如下:

#include <iostream>
using namespace std;


class String
{
public:
 String(char* str = "")
 :_str(new char[strlen(str)+1])
 {
 strcpy(_str, str);
 }

 //传统写法,开辟空间
 String(const String& s)
 {
 _str = new(char[strlen(s._str) + 1]);
 strcpy(_str, s._str);
 }

 //现代写法,利用构造函数
 //String(const String& s)
 // :_str(NULL)
 //{
 // String tmp(s._str);
 // swap(_str, tmp._str);
 //}

 //****************赋值运算符重载**************
 //String& operator=(const String& s)
 //{
 // if (this != &s)
 // {
 // delete[] _str;
 // _str = new char[strlen(s._str) + 1];
 // strcpy(_str, s._str);
 // }
 // return *this;
 //}

 //****************赋值运算符重载**************
 String& operator=(String& s)
 {
 swap(_str, s._str);
 return *this;
 }

 //***************析构函数********************
 ~String()
 {
 if (_str)
 {
  delete[] _str;
 }
 }

private:
 char* _str;
};

二:写时拷贝

写时拷贝:引入一个计数器,每片不同内容的空间上都再由一个计数器组成,在构造第一个类指向时,计数器初始化为1,之后每次有新的类也指向同一片空间时,计数器加价;在析构时判断该片空间对应计数器是否为1,为1则执行清理工作,大于1则计数器-1。如果有需要进行增删等操作时,再拷贝空间完成,有利于提高效率。

写法一:

#include <iostream>
using namespace std;

class String
{
public:
 String(char* str = "")
 :_str(new char[strlen(str)]+1)
 , _refCount(new int(1))
 {
 strcpy(_str, str);
 }

 String(const String& str)
 : _str(str._str)
 ,_refCount(str._refCount)
 {
 (*_refCount)++;
 }

 ~String()
 {
 release();
 }

 String& operator= (const String& s)
 {
 if (_str != s._str)
 {
  release();
  _refCount = s._refCount;
  (*_refCount)++;
  _str = s._str;
 }
 return *this;
 }

 void release()
 {
 if ((*--_refCount) == 0)
 {
  delete[] _str;
  delete _refCount;
 }
 }

private:
 char* _str;
 int* _refCount;
};

缺点:每构造一个新类,就会多开四个字节,会导致空间中有许多的内存碎片。

第二种:

class String
{
public:
 String(char* str = "")
 :_str(new char[strlen(str)+1+4])
 {
 *(int*)_str = 1;
 _str += 4;
 strcpy(_str, str);
 }

 String(const String& s)
 :_str(s._str)
 {
 ++GetCount();
 }

 ~String()
 {
 release();
 }

 String& operator=(const String& s)
 {
 if (this != &s)
 {
  realease();
  _str = s._str;
  GetCount()++;
 }
 return *this;
 }

 void release()
 {
 if (--GetCount() == 0)
 {
  _str -= 4;
  delete[] _str; 
 }
 }

 int& GetCount()
 {
 return *((int*)_str - 1);
 }

private:
 char* _str;
};

注意:由于计数器存放在了_str首地址-4的地址上,所以在析构时一定要注意全部释放,避免内存泄漏。

图示如下:

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对的支持


# c  # 深浅拷贝  # 写时拷贝  # 深拷贝  # C/C++ 浅拷贝和深拷贝的实例详解  # C++拷贝构造函数(深拷贝与浅拷贝)详解  # 详解C++中String类模拟实现以及深拷贝浅拷贝  # C++深拷贝与浅拷贝的区别及应用  # C++浅拷贝与深拷贝及引用计数分析  # C++中浅拷贝与深拷贝的详解及其作用介绍  # C++写时拷贝实现原理及实例解析  # 深入理解C/C++中的写时拷贝  # 掌握C++:揭秘写时拷贝与浅深拷贝之间的关系  # 运算符  # 就会  # 放在  # 也会  # 相关内容  # 第一个  # 中有  # 给大家  # 并将  # 这篇文章  # 上都  # 谢谢大家  # 多说  # 第二种  # 再由  # 将原  # 提高效率  # 有疑问  # align  # realease 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 今日头条AI怎样推荐抢票工具_今日头条AI抢票工具推荐算法与筛选【技巧】  php静态变量怎么调试_php静态变量作用域调试技巧【解答】  西安专业网站制作公司有哪些,陕西省建行官方网站?  edge浏览器无法安装扩展 edge浏览器插件安装失败【解决方法】  网站建设要注意的标准 促进网站用户好感度!  Laravel怎么实现验证码(Captcha)功能  微信小程序 闭包写法详细介绍  html5的keygen标签为什么废弃_替代方案说明【解答】  javascript和jQuery中的AJAX技术详解【包含AJAX各种跨域技术】  移动端脚本框架Hammer.js  制作ppt免费网站有哪些,有哪些比较好的ppt模板下载网站?  Laravel如何记录日志_Laravel Logging系统配置与自定义日志通道  WEB开发之注册页面验证码倒计时代码的实现  实现点击下箭头变上箭头来回切换的两种方法【推荐】  如何快速生成专业多端适配建站电话?  如何在腾讯云服务器上快速搭建个人网站?  原生JS获取元素集合的子元素宽度实例  最好的网站制作公司,网购哪个网站口碑最好,推荐几个?谢谢?  Python自动化办公教程_ExcelWordPDF批量处理案例  Chrome浏览器标签页分组怎么用_谷歌浏览器整理标签页技巧【效率】  Laravel路由Route怎么设置_Laravel基础路由定义与参数传递规则【详解】  html5怎么画眼睛_HT5用Canvas或SVG画眼球瞳孔加JS控制动态【绘制】  LinuxCD持续部署教程_自动发布与回滚机制  如何自定义safari浏览器工具栏?个性化设置safari浏览器界面教程【技巧】  Laravel Artisan命令怎么自定义_创建自己的Laravel命令行工具完全指南  打开php文件提示内存不足_怎么调整php内存限制【解决方案】  Laravel如何正确地在控制器和模型之间分配逻辑_Laravel代码职责分离与架构建议  如何用狗爹虚拟主机快速搭建网站?  Win11怎么查看显卡温度 Win11任务管理器查看GPU温度【技巧】  EditPlus中的正则表达式 实战(1)  Laravel如何实现事件和监听器?(Event & Listener实战)  高配服务器限时抢购:企业级配置与回收服务一站式优惠方案  高端建站如何打造兼具美学与转化的品牌官网?  Bootstrap整体框架之JavaScript插件架构  北京网站制作公司哪家好一点,北京租房网站有哪些?  Laravel怎么配置自定义表前缀_Laravel数据库迁移与Eloquent表名映射【步骤】  如何在阿里云完成域名注册与建站?  PHP正则匹配日期和时间(时间戳转换)的实例代码  Laravel如何使用缓存系统提升性能_Laravel缓存驱动和应用优化方案  微信小程序 wx.uploadFile无法上传解决办法  Laravel如何使用Guzzle调用外部接口_Laravel发起HTTP请求与JSON数据解析【详解】  CSS3怎么给轮播图加过渡动画_transition加transform实现【技巧】  Laravel如何实现用户密码重置功能?(完整流程代码)  Laravel怎么在Controller之外的地方验证数据  Laravel如何使用Blade组件和插槽?(Component代码示例)  如何在局域网内绑定自建网站域名?  如何在云指建站中生成FTP站点?  详解Android图表 MPAndroidChart折线图  制作企业网站建设方案,怎样建设一个公司网站?  哪家制作企业网站好,开办像阿里巴巴那样的网络公司和网站要怎么做?