cpp primer笔记100-拷贝控制

news2024/11/24 11:03:48
  1. 如果拷贝构造函数如果传递的参数不是引用类型,则调用拷贝永远不成功,因为如果调用了拷贝构造函数,则必须拷贝它的实参,但是为了拷贝实参,我们又需要调用拷贝构造函数,如此循环。

  2. 如果想要删除默认构造函数,默认拷贝构造函数,默认赋值运算符,可一直接在声明函数后面加上=delete,但是析构函数最好不要被删除,否则无法释放对象内存。

    #include<string>
    #include<cstring>
    #include<cstring>
    #include<iostream>
    #include<algorithm>
    #include<functional>
    #include<unordered_set>
    #include<unordered_map>
    using namespace std;
    class numbered
    {
    public:
    	int a, b, c;
    	numbered() = delete;
    	numbered(int x, int y, int z) :a(x), b(y), c(z) {};
    };
    signed main()
    {
    	numbered num1(1, 2, 3);
    	//numbered num2; 默认构造函数被删除无法使用
    	
    }
    

    ![[Pasted image 20230923211325.png]]

  3. shared_ptr的引用计数:
    ![[Pasted image 20230923220043.png]]

  4. 由于右值引用只能绑定到临时对象,所以右值引用的对象将要被销毁,而且该对象没有其他变量使用。如果想让一个左值转换为对应的右值引用类型,可以通过std::move函数来调用,右值引用注意一下几点:

    • 一旦完成资源的移动,移后原对象处于销毁无害状态,原对象必须不在指向被移动的资源,这些资源所有权已经归属新创建的对象。

    • 如果一个函数被noexcept修饰,则如果这个函数抛出异常,则程序无论能否被catch捕获会中断执行,而没有被该关键词修饰的函数则可能被catch捕获然后继续执行catch和catch后面的代码。移动构造函数需要notexcept修饰的原因是,如果在对vector进行push_back重新分配空间时,移动失败的话vector内的元素会发生改变,可能已经不存在,导致程序后面发生错误,拷贝失败的话,vector旧元素并不发生改变,vector原有元素还是会存在的。

    • 移动构造函数基本是针对指针进行操作的,移动是将原有的指针指向的位置传递给新的指针,并且将原有的指针置空,拷贝是将原有的指针指向的数据copy一份给新的指针,所以移后原对象基本可以说是报废不可用,所以说不建议对移后原对象进行操作。当然将指针置空还有另一个原因,就是如果两个对象的指针都指向同一个地址,则会导致同一个地址的指针被delete多次导致报错,所以需要将指针置空

    • ![[Pasted image 20230924130706.png]]

    • 当一个类既有移动构造函数,又有拷贝构造函数的时候,如果传入的参数是非静态右值,则使用移动构造函数,否则使用拷贝构造函数,如果一个类只有拷贝构造函数,则包括右值在内的所有类型参数使用拷贝构造函数。

    #define _CRT_SECURE_NO_WARNINGS 1
    #include <iostream>
    #include <fstream>
    #include <array>
    #include <vector>
    #include <string>
    #include <exception>
    #include <algorithm>
    #include <deque>
    #include <numeric>
    #include <memory>
    #include <initializer_list>
    #include <exception>
    #include <cstring>
    class StrVec
    {
    private:
    	char* str = nullptr;
    	int len = 0;
    public:
    	StrVec(const char* s)
    	{
    		len = std::strlen(s);
    		str = new char[len + 1];
    		std::strcpy(str, s);
    	}
    	StrVec(const StrVec& s)
    	{
    		if (s.str != nullptr)
    		{
    			len = s.len;
    			delete str;
    			str = new char[len + 1];
    			std::strcpy(str, s.str);
    		}
    	}
    	StrVec(StrVec&& s) noexcept
    	{
    		std::cout << "调用移动构造函数" << std::endl;
    		if (s.str != nullptr)
    		{
    			len = s.len;
    			delete str;
    			str = s.str;
    			s.str = nullptr;
    		}
    	}
    	StrVec& operator=(const StrVec& s)
    	{
    		if (this != &s)
    		{
    			delete str;
    			len = s.len;
    			str = new char[len + 1];
    			std::strcpy(str, s.str);
    		}
    		return *this;
    	}
    	StrVec& operator=(StrVec&& s) noexcept
    	{
    		throw "werwer";
    		if (this != &s)
    		{
    			delete str;
    			len = s.len;
    			str = s.str;
    			s.str = nullptr;
    		}
    		return *this;
    	}
    	friend std::ostream& operator<<(std::ostream& os, const StrVec& val)
    	{
    		os << val.len << " " << val.str;
    		return os;
    	}
    };
    int main()
    {
    	//int&& rr1 = 42;//正确:字面常量是右值
    	int&& rr2 = rr1; 错误不能将一个右值引
    	用绑定到一个右值引用类型的变量上
    	//int&& rr3 = std::move(rr1);//可以通过move函数来获取左值上的右值引用
    	StrVec a = "aaaaa";
    	StrVec b = "bbbbb";
    	StrVec c = "ccccc";
    	StrVec d = "ddddd";
    	try
    	{
    		std::cout << a << " " << b << " " << c << " " << d << std::endl;
    		b = a;
    		StrVec e = a;
    		std::cout << a << " " << b << " " << c << " " << d << " " << e << std::endl;
    		StrVec f = std::move(b);
    		d = std::move(e);
    		std::cout << a << " " << c << " " << d << " " << " " << f << std::endl;
    	}
    	catch (...)
    	{
    		std::cout << "抛出异常";
    	}
    	return 0;
    }
    
  5. 多个功能类似的函数可以组合成一个函数表,便于调用,但是使用关联容器作为函数表可能无法存入函数类对象,所以引入function函数来出入对象

    #define _CRT_SECURE_NO_WARNINGS 1
    #include<iostream>
    #include<map>
    struct  divide
    {
    	int operator()(int denomiator, int divisor)
    	{
    		return denomiator / divisor;
    	}
    	divide(int) {};
    };
    int add(int i, int j) { return i + j; }
    int main()
    {
    	auto mod = [](int i, int j) {return i % j; };
    	std::map<std::string, int(*)(int, int)> binops;
    	binops.insert({ "+",add });
    	binops.insert({ "%",mod });
    	//lambda在没有捕获任何变量的情况下赋值为函数指针,而不是函数闭包
    	//如果在捕获情况下传入则会编译错误
    	//binops.insert({ "/",divide }); 错误
    	std::cout << binops["%"](10, 4);
    	return 0;
    }
    
    
    2
    
#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
#include<map>
#include<functional>
#include<algorithm>
struct divide
{
	int operator()(int denomiator, int divisor)
	{
		return denomiator / divisor;
	}
};
int add(int i, int j) { return i + j; }
int func(int i, int j) { return 2 * i + j; }
int func(int i, int j, int k) { return 3 * i + 2 * j + k; }
int main()
{
	auto mod = [](int i, int j) { return i % j; };
	std::map<std::string, std::function<int(int, int)>> binops;
	binops.insert({ "+",add });
	binops.insert({ "/",divide() });
	binops.insert({ "-",std::minus<int>() });
	binops.insert({ "*",[](int i,int j) {return i * j; } });
	binops.insert({ "%",mod });
	//binops.insert({ "func",func }); 错误不能直接写重载函数名
	binops.insert({ "func",[](int i,int j) {return func(i,j); } });//通过lambda代替
	std::ostream_iterator<int> os(std::cout, "\n");
	os = binops["+"](10, 5);
	os = binops["/"](10, 5);
	os = binops["-"](10, 5);
	os = binops["*"](10, 5);
	os = binops["%"](10, 5);
	return 0;
}

15
2
5
50
0

![[Pasted image 20230924155429.png]]

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1067591.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【浅谈IDE宏指令录制】为加速chrome扩展国际化,我从vscode回归notepad++

vscode 的宏录制功能 —— 差强人意 安装vscode开源扩展&#xff1a;https://github.com/C10udburst/macros-vscode.git 可开启类似于 notetepad 的宏录制与回放功能&#xff01;比如录制字符串替换&#xff0c;能记录操作之时&#xff0c;替换对话框中的文本&#xff01;&am…

Mac上安装Java的JDK多版本管理软件jEnv

JDK的多版本管理软件主要有以下三种&#xff1a; jEnv jEnv 是一个命令行工具&#xff0c;可以帮助您管理和切换不同版本的 Java 环境。它可以让您在不同的项目之间轻松切换 Java 版本。您可以使用 jenv global 命令设置全局 Java 版本&#xff0c;也可以使用 jenv local 命令…

【力扣】42. 接雨水

这道题我卡了差不多1个小时&#xff0c;不是不会做&#xff0c;是不知道怎么能用栈来实现&#xff0c;后面看了一个博主的视频&#xff0c;豁然开朗&#xff0c;我主要的纠结点在于当指针指到7的时候&#xff0c;我计算出4到7的水块是2&#xff0c;但实际上是0&#xff0c;因为…

打造完美家庭空间,让生活更加舒适

在现代繁忙的都市生活中&#xff0c;家是人们温暖而舒适的避风港。而如何打造一个恰到好处的家庭空间&#xff0c;成为了许多人心中的追求。今天&#xff0c;就让我们来探索一些空间布局方案&#xff0c;为您的家庭营造一个完美融合功能与美感的舒适空间。 &#x1f3e0;&…

RT-Thread 内存管理(学习二)

内存堆管理应用示例 这是一个内存堆的应用示例&#xff0c;这个程序会创建一个动态的线程&#xff0c;这个线程会动态申请内存并释放&#xff0c;每次申请更大的内存&#xff0c;当申请不到的时候就结束。 #include <rtthread.h>#define THREAD_PRIORITY 25 #defi…

微信小程序通过createSelectorQuery获取元素 高度,宽度与界面距离

小程序官方有提供给我们一个 const query wx.createSelectorQuery() 函数 我们可以先编写这样一段代码 wxml <view><button bindtap"getDom">点击查看</button><view class "textIn" style "height: 100px;width: 30px;&quo…

IDEA 初始化配置

进入所有配置 配置外观 取消退出前确认和启动软件自动打开之前最后的项目 设置使用键盘Ctrl鼠标滚轮调整代码大小 设置字符集 设置构建工具 自动构建与自动编译 启用注解

think-rom(thinkphp) 数据库参数绑定bigint被处理成浮点数(科学计数)问题

背景 think-orm(db库)在处理长整型参数时会进行一次安全处理转换成浮点型&#xff0c;但php浮点数超过一定长度就会失去精准度变成科学计数&#xff0c;导致where条件匹配不上&#xff0c;这个问题目前覆盖thinkphp3到8全系&#xff1b; 解决方案 全局搜索 (float) $val[0] : …

迁移人大金仓问题汇总

迁移人大金仓问题汇总 1、SQL 错误 [42883]: 错误: 操作符不存在: tinyint Hint: 没有匹配指定名称和参数类型的操作符. 您也许需要增加明确的类型转换. SELECT id, model_id, process_definition_id, task_definition_key, TYPE, OPTIONS, scope_userid, scope_clas…

雷达分辨率单元、单向/双向雷达方程、天气雷达方程简介

一、点状目标 如果两个点状目标在一个分辨率单元中,经典脉冲雷达只能看到一个目标。 点状目标 二、雷达距离分辨率 对于简单的键控开/关脉冲调制: 对于使用脉冲内调制的雷达,距离分辨率取决于压缩脉冲的脉冲持续时间。脉冲压缩比(PCR)取决于传输带宽BWtx,即距离分辨率取…

大语言模型学到什么

背景&#xff1a; 这篇文章是对《LANGUAGE MODELS REPRESENT SPACE AND TIME》论文的翻译加解读。之所以选这篇文章是因为最近在研究大模型的可解释性&#xff0c;以及基于可解释性对大模型的下游任务适配做训练级别可控性增强研究。其实总结成两句话就是&#xff1a; 1.大模…

amber14自由能计算及增强采样方法

1分子力学简介 1.1分子力学的基本假设 1.2分子力学的主要形式 2分子力场 2.1分子力场的简介 2.2分子力场的原理 分子力场的分类及应用 二.LINUX入门 教学目标&#xff1a;掌握数值计算平台&#xff0c;熟悉计算机语言&#xff0c;能够使用vim编辑器简单编辑文件。 3LINUX 简介 …

84.柱状图中最大的矩形

class Solution { public:int largestRectangleArea(vector<int>& heights) {//法一&#xff1a;枚举宽&#xff0c;超时if(0){int nheights.size();int area0;for(int left0;left<n;left){int minHINT_MAX;for(int rightleft;right<n;right){minHmin(minH,hei…

Spring实例化源码解析之MessageSource(七)

前言 在阅读完 registerBeanPostProcessors 源码之后&#xff0c; 下一步就进入到 initMessageSource&#xff0c;这一步主要作用是初始化国际化文件。 源码分析 这段源码是一个Java方法&#xff0c;用于初始化消息源&#xff08;MessageSource&#xff09;。在Spring框架中…

万字详解HTTP协议面试必备技能

目录 一、HTTP 是什么 二、理解 "应用层协议" 2.1理解 HTTP 协议的工作过程 2.2HTTP 协议格式 2.3抓包工具的使用 2.4抓包工具的原理 2.5抓包结果 2.5.1HTTP请求 2.5.2HTTP响应 2.6协议格式总结 三、HTTP 请求 (Request) 3.1认识 URL 3.1.1URL 基本格式 …

Jmeter 多实例压测

Apache JMeter 是一个开源的 Java 应用程序&#xff0c;用于性能测试和负载测试。它最初是为测试 Web 应用程序而创建的&#xff0c;但现在已广泛用于测试各种不同类型的应用程序&#xff0c;包括数据库、消息队列、FTP 服务器等。JMeter 提供了丰富的功能&#xff0c;使您能够…

行内元素文字背景被截断的问题,如何进行修改?

1.问题如图所示 这里背景是被截断的&#xff0c;有什么css样式可以修改它呢&#xff1f; 效果实现如图&#xff1a; 不一样的地方在于这里的截断位置是引用了开头和结尾的位置。 如何实现&#xff1f; <!DOCTYPE html> <html lang"en"> <head><…

终于找到了!Jenkins最全最详细入门指南

目录 一、介绍 二、功能 三、搭建Jenkins目的 四、安装 五、Jenkins的三大概念 六、创建job & 配置git 七、构建触发器&#xff08;build triggers&#xff09; 八、运行python脚本 九、测试报告展示 十、配置邮件并发送 最后 一、介绍 Jenkins是开源的,使用Java编…

Linux系统部署PostgreSQL 单机数据库

安装方式 1 安装包方式 &#xff08;Packages and Installers&#xff09; 支持的操作系统包括 liunxMacosWindowsBSDSolaris 2 源码安装 &#xff08;Source code&#xff09; 下载源码包 通过下载地址PostgreSQL: File Browser 可以看到有各个版本的源码目录 选择13.1…

二维离散傅里叶变换的实现

二维离散傅里叶变换的实现 1.使用Python包实现2.使用c实现2.1 FFTW库安装2.2 结果比较 参考文献 1.使用Python包实现 import numpy as np import matplotlib.pyplot as plt anp.array([0, 2, 4, 1,6, 1, 3, 2,5]).reshape(3,3) fnp.fft.fft2(a) fshiftnp.fft.fftshift(f) mag…