C++之拷贝控制 (Copy Control)
只有2种成员
值成员;
指针成员; 依实现可分为raw pointer / shared_ptr;

现在,仅考虑第③种:资源对象共享 角度来考虑拷贝控制
类的两种语义:值语义、似指针

编译器提供的default版本的copy constructor/ copy assignment的语义:
0. 默认构造:对每个成员进行默认:① 内置类型、指针类型 若未指定初始值则其值未定义。 ② T类类型成员采用T类型的默认构造。
1. 拷贝构造: 对rhs的每个成员进行拷贝。(指针成员只拷贝指针值,不进行其指向的资源对象的拷贝)
2. 拷贝赋值:修改左侧instance的各成员值为右侧对象的对应成员值,即:对lhs的每个成员进行 lhs.member = rhs.member 赋值。
实现自带的 引用计数器
实现机制:
指向同一资源对象instance的多个shared_ptr 联系着同一个 “该资源对象instance的引用计数器”instance
【1个引用计数器实例,针对的肯定是 1个资源instance】
当shared_ptr创建时,引用计数1;拷贝时+1;销毁时-1、并检查:若引用计数变为0,进行资源的释放。
自行实现:
类HasPtr通过 *_p 持有一个string对象资源。
(若该string对象资源 为多个HasPtr的instance-s所共享,则HasPtr的这多个instance-s间共同维护一个的“该string instance的引用计数器”)
shared_ptr<Resource> = {
Resource* pt;
int* referCount;
}

4种可能的 拷贝构造/拷贝赋值/析构 方案
若对某个成员是“值副本持有”:( {T* _pt; T instance} 视作一体 )
copy constructor:拷贝该部分成员资源
T: t (rhs.t); // 默认行为
*pt: pt = new T(*rhs.t);
shared_ptr: sp = make_shared<T>(*rhs.sp);
copy = :拷贝该部分成员资源;释放原资源
T: t = rhs.t; // 默认行为
*pt: T* newpt = new T(*rhs.t); delete pt; pt = newpt; // 可优化吧(指针值相等则不拷贝)
shared_ptr: sp = make_shared<T>(*rhs.sp); // sp.reset(new T(*rhs.sp));也行吧?
move constructor:接管临时对象的instance资源
T: t( move(rhs.t))
*pt: pt = rhs.pt; rhs.pt = nullptr;
shared_ptr: sp(rhs.sp) // 默认行为
move = : 接管临时对象的资源
T: t = move(rhs.t); // 匹配T instance的move assignment
*pt: delete pt; pt = rhs.pt; rhs.pt = nullptr;
shared_ptr: sp = rhs.sp; // 默认行为
析构:
T: // 默认即可
*pt: delete pt;
shared_ptr: // 默认即可
=====================================================================
若对某个成员属于“作为引用者之一”:(成员不可能是 T t; 形式)
copy constructor:指针绑上
*pt: pt( rhs.pt); // 默认
shared_ptr: sp( rhs.sp); // 默认
copy = : 指针绑上
*pt: pt = rhs.pt; // 默认
shared_ptr: sp = rhs.sp; // 默认
move constructor:
*pt: pt (rhs.pt); //默认
shared_ptr: sp (rhs.sp) // 默认
move = :
*pt: pt = rhs.pt; // 默认
shared_ptr: sp = rhs.sp; // 默认
析构:
*pt: //默认即可。析构函数中不能写定delete pt; 需要在最后一个对象使用结束后 显式释放
shared_ptr: // 默认
C++之拷贝控制 (Copy Control)的更多相关文章
- C/C++:copy control (拷贝控制)
前言:当定义一个类的时候,我们显示或者隐式地指定在此类型的对象拷贝,移动,赋值,销毁时做些什么,一个类通过定义五种特殊的成员函数来控制这些操作,包括拷贝构造函数,拷贝赋值运算符,移动构造函数,移动赋值 ...
- [c++] Copy Control
C++ allows the programmer to define how objects are to be copied, moved, assigned and destroyed. Tog ...
- C++的那些事:类的拷贝控制
1,什么是类的拷贝控制 当我们定义一个类的时候,为了让我们定义的类类型像内置类型(char,int,double等)一样好用,我们通常需要考下面几件事: Q1:用这个类的对象去初始化另一个同类型的对象 ...
- C++ Primer : 第十三章 : 拷贝控制之拷贝、赋值与销毁
拷贝构造函数 一个构造函数的第一个参数是自身类类型的引用,额外的参数(如果有)都有默认值,那么这个构造函数是拷贝构造函数.拷贝构造函数的第一个参数必须是一个引用类型. 合成的拷贝构造函数 在我们没 ...
- c/c++ 拷贝控制 构造函数的问题
拷贝控制 构造函数的问题 问题1:下面①处的代码注释掉后,就编译不过,为什么??? 问题2:但是把②处的也注释掉后,编译就过了,为什么??? 编译错误: 001.cpp: In copy constr ...
- c/c++ 拷贝控制 右值与const引用
拷贝控制 右值与const引用 背景:当一个函数的返回值是自定义类型时,调用侧用什么类型接收?? 1,如果自定义类型的拷贝构造函数的参数用const修饰了:可以用下面的方式接收. Test t2 = ...
- 【C++ Primer | 15】构造函数与拷贝控制
合成拷贝控制与继承 #include <iostream> using namespace std; class Base { public: Base() { cout << ...
- 零拷贝-zero copy
Efficient data transfer through zero copy Zero Copy I: User-Mode Perspective 0. 前言 在阅读RocketMQ的官方文档时 ...
- C/C++基础----拷贝控制
拷贝控制操作,有5个特殊成员函数copy ctor,copy =opt,move ctor,move =opt,dtor 有哪些地方会用到 拷贝初始化 除了=定义变量时 参数传递和函数返回时 花括号列 ...
随机推荐
- system.exit(int status)中status值不同时的区别
status为0时为正常退出程序,也就是结束当前正在运行中的java虚拟机. status为非0的其他整数(包括负数,一般是1或者-1),表示非正常退出当前程序. 可以明确的是,无论status是什么 ...
- 51nod 1594 Gcd and Phi(莫比乌斯反演)
题目链接 传送门 思路 如果这题是这样的: \[ F(n)=\sum\limits_{i=1}^{n}\sum\limits_{j=1}^{n}\phi(gcd(i,j)) \] 那么我们可能会想到下 ...
- opencv 程序
IplImage结构中的一个元素:struct _IplROI *roi; //图像感兴趣区域,当该值非空时,只对该区域进行处理 . ROI :Region of Interest,表示感兴趣的区 ...
- 201871010123-吴丽丽《面向对象程序设计(Java)》第四周学习总结
201871010123-吴丽丽<面向对象程序设计(Java)>第四周学习总结 项目 内容 这个作业属于哪个课程 https://www.cnblogs.com/nwnu-daizh/ 这 ...
- conda管理python环境
https://blog.csdn.net/wld914674505/article/details/80615761 source activate python36
- .Net反射-基础2-BindingFlags参数
BindingFlags参数用于指定反射查找的范围在调用下列方法时会用到BindingFlags参数 // 调用方法. InvokeMethod // 创建实例. CreateInstance // ...
- vue大文件上传插件选哪个好?
文件夹数据库处理逻辑 public class DbFolder { JSONObject root; public DbFolder() { this.root = new JSONObject() ...
- cube.js 最近的一些更新
cube.js 是一个和不错的数据分析框架,最近又有了一些新的功能支持,以下是一些简单的 总结 基于web socket 的预览支持 react hooks api 支持 支持基于reecharts ...
- pychram-redis破解
1. Preferences -> Plugins-> 选择右下角Browse repositories 2. 搜索Iedis 3. 找到Iedis插件目录:C:\Users\用户名\.P ...
- hibernate关联关系 (多对多)
hibernate的多对多 hibernate可以直接映射多对多关联关系(看作两个一对多 多对多关系注意事项 一定要定义一个主控方 多对多删除 主控方直接删除 被控方先通过主控方解除多对多关系,再删 ...