C++的内存管理和模板

news2024/11/17 16:27:34

文章目录

  • 一、内存管理
    • 1.内存的分布
    • 2.C++中的动态内存管理
    • 3.重载new和重载delete
    • 4.new和delete的实现原理
    • 5.定位new
  • 二、模板
    • 1.泛型编程
    • 2.函数模板
      • 1.定义模板
      • 2.实例化模板
      • 3.模板类型的参数
      • 4.非类型模板参数
    • 3.类模板
      • 1.定义模板
      • 2.实例化模板
      • 3.模板的成员函数
  • 总结

一、内存管理

1.内存的分布

在这里插入图片描述
这是在C语言中的内存分布,C++的内存分布也是这样。下面让我们看看C++中的动态内存开辟是什么样的吧。

2.C++中的动态内存管理

C语言内存管理方式在C++中可以继续使用,但有些地方就无能为力,而且使用起来比较麻烦,因此C++又提出了自己的内存管理方式:通过new和delete操作符进行动态内存管理。

int main()
{
	int* ptr1 = new int;      //动态申请一个int类型的空间	
	char* cptr = new char;    //动态申请一个char类型的空间	
	int* ptr2 = new int(10);  //动态申请一个int类型的空间并初始化为10
	int* ptr3 = new int[10];  //动态申请10个int类型的空间
	delete ptr1;	//释放ptr1的空间
	delete cptr;	//释放ptr1的空间
	delete ptr2;	//释放ptr2的空间
	delete[] ptr3;	//释放ptr3的空间
	return 0;
}

注意:申请和释放单个元素的空间,使用new和delete操作符,申请和释放连续的空间,使用new[]和delete[]。
操作案例:

class A
{
public:
	A(int a = 0) :a(a)
	{
		cout << "A():" << this << endl;
	}
	~A()
	{
		cout << "~A():" << this << endl;
	}
private:
	int a;
};
int main()
{
	cout << "开始创建1个A类型的对象:" << endl;
	A* ptr1 = new A;
	cout << endl;
	cout << "开始创建3个A类型的对象:" << endl;
	A* ptr2 = new A[3];
	cout << endl;
	cout << "开始释放1个A类型的对象:" << endl;
	delete ptr1;
	cout << endl;
	cout << "开始释放3个A类型的对象:" << endl;
	delete[] ptr2;
	return 0;
}

在这里插入图片描述
new/delete 和 malloc/free最大区别是 new/delete对于[自定义类型]除了开空间还会调用相应的构造函数和析构函数。而内置类型几乎一样。

3.重载new和重载delete

某些应用程序对内存分配有特殊的需求,因此我们无法将标准内存管理机制直接应用于这些程序,它们常常需要自定义内存分配的细节,比如使用关键字new将对象放置在特定的内存空间中。为了实现这一目的,应用程序需要重载new运算符和delete运算符以控制内存分配过程。
尽管我们说重载new和delete,但实际上重载这两个运算符和其他运算符重载大不相同。

int main()
{
	A* ptr1 = new A(10);//分配并初始化一个A对象
	A* ptr2 = new A[10];//分配10个默认初始化的A对象
	delete ptr1;//销毁 *ptr1,并释放ptr1指向的空间
	delete[] ptr2;//销毁数组中的元素,然后释放相应的空间
	return 0; 
}

`在这里插入图片描述
上述代码实际执行三步操作:
1.new表达式调用一个名为 operator new(或者opreator new[])的标准库函数。该函数分配一块足够大的,原始的,未命名的内存空间以便存储特定对象(或者对象数组)。
2.编译器运行相应的构造函数以构造这些对象,并传入初始值。
3.对象被分配了空间并且完成了构造,返回一个指向该对象的指针。
在这里插入图片描述
上述代码实际执行两步操作:
1.对ptr1所指向的对象或者ptr2所指向的数组中的元素执行相应的析构函数。
2.编译器调用名为 operator delete(或者opreator delete[])的标准库函数进行释放空间。
总结:
new和delete是用户进行动态内存申请和释放的操作符,operator new 和operator delete是系统提供的全局函数,new在底层调用operator new全局函数来申请空间,delete在底层通过operator delete全局函数来释放空间。
在这里插入图片描述
这是在VS上找到的重载new和delete的函数原型。
通过上述两个全局函数的实现知道,operator new 实际也是通过malloc来申请空间,operator delete 最终是通过free来释放空间的。
注意:当自定义了全局的operator new 函数和operator delete函数后,我们就担负起了控制动态内存分配的职责。这两个函数必须是正确的,因为它们是程序整个处理过程中至关重要的一部分。

4.new和delete的实现原理

对于内置类型来说:new和malloc,delete和free基本类似,不同的地方是:new/delete申请和释放的是单个元素的空间,new[]和delete[]申请的是连续空间,而且new在申请空间失败时会抛异常,malloc会返回NULL。
对于自定义类型类型来说:
new调用operator new函数申请空间,并且在申请的空间上执行构造函数,完成对象的构造。
new [ ] 调用operator new[]函数,在operator new[]中实际调用operator new函数完成N个对象空间的申请,并且在申请的空间上执行构造函数,完成对象的构造。
delete先进行析构函数,完成对象中资源的清理工作,在调用operator delete函数释放对象的空间。
delete [ ]先进行相应次数的析构函数,完成对象中资源的清理工作, 调用operator delete[]释放空间,实际在operator delete[]中调用operator delete来释放空间。
注意:即使是底层调用的malloc和free,我们也不能使用new申请空间用free释放或者使用malloc申请空间使用delete释放。我们不可以混合使用,因为在一些特定的场景下会出现意想不到的结果。

5.定位new

定位new表达式是在已分配的原始内存空间中调用构造函数初始化一个对象。
定位new的形式如下:

new (place_address) type
new (place_address) type (initializers)
new (place_address) type [size]
new (place_address) type[size]{ barced initializer - list }

place_address必须是一个指针,同时在initializers中提供一个(可能未空的)以逗号分割的初始值列表,该初始值列表将用于构造新分配的对象。

int main()
{	
	//ptr现在指向的只不过是与A对象相同大小的一段空间,还不能算是一个对象,因为构造函数没有执行
	A* ptr = (A*)malloc(sizeof(A));  
	//注意:如果A类的构造函数有参数时,此处需要传参
	new(ptr)A;	
	ptr->~A();
	free(ptr);
	return 0;
}

定位new表达式在实际中一般是配合内存池使用。因为内存池分配出的内存没有初始化,所以如果是自定义类型的对象,需要使用new的定义表达式进行显示调构造函数进行初始化。

二、模板

面对对象编程(OOP)和泛型编程都能处理在编写程序时不知道类型的情况。不同之处在于:OOP能处理类型在程序运行之前都未知的情况,而在泛型编程中,在编译时就能获知类型了。
尽量使用初始化列表初始化,因为不管你是否使用初始化列表,对于自定义类型成员变量,一定会先使用初始化列表初始化。

1.泛型编程

编写与类型无关的通用代码,是代码复用的一种手段。模板是泛型编程的基础。一个模板就是一个创建类或函数的蓝图或者说公式

2.函数模板

1.定义模板

假如我们需要编写一个函数来比较两个值,并且指出实际的关系,在实际中,我们可能想要定义多个函数,通过函数重载来比较每一种类型。
例如:

int Compare(const int a,const int b)
{
	if (a < b)
	{
		return -1;
	}
	if (a == b)
	{
		return 0;
	}
	else
	{
		return 1;
	}
}
int Compare(const char a, const char b)
{
	if (a < b)
	{
		return -1;
	}
	if (a == b)
	{
		return 0;
	}
	else
	{
		return 1;
	}
}
int main()
{
	int a = 20;
	int b = 10;
	char cha = 'a';
	char chb = 'b';
	cout << Compare(a, b) << endl;
	cout << Compare(cha, chb) << endl;
	return 0;
}

在这里插入图片描述
当我们要比较浮点数在重载一个浮点数类型的比较,这样做太麻烦且逻辑一样,代码冗余。
函数模板格式:

template<typename T1, typename T2, ..., typename Tn>
返回值类型 函数名(参数列表) {}

下面我们把它改为模板来观察一下结果吧:

template<typename T>
int Compare(const T a, const T b)
{
	if (a < b)
	{
		return -1;
	}
	if (a == b)
	{
		return 0;
	}
	else
	{
		return 1;
	}
}
int main()
{
	int a = 20;
	int b = 10;
	char cha = 'a';
	char chb = 'b';
	float fa = 1.25;
	float fb = 1.25;
	cout << Compare(a, b) << endl;
	cout << Compare(cha, chb) << endl;
	cout << Compare(fa, fb) << endl;
	return 0;
}

在这里插入图片描述
我们只是通过一个函数就解决了我们整形,字符和浮点型的比较。

2.实例化模板

当我们调用一个函数模板时,编译器(通常)用函数实参来为我们推断模板实参,当我们调用Compare时,编译器使用实参的类型来确定绑定到模板参数T的类型。
例如:

cout << Compare(a, b) << endl;//T的类型为int

我们的实际参数为int,编译器会推断出模板实参为int,并将它绑定到模板参数T,编译器用推断出的模板参数来为我们实例化一个特定版本的函数。

//实例化出int Compare(const int a, const int b)
cout << Compare(a, b) << endl;
//实例化出int Compare(const char a, const char b)
cout << Compare(cha, chb) << endl;

这些编译器生成的版本通常被称为模板的实例。

3.模板类型的参数

我们的Compare函数有一个模板类型参数,一般来说我们可以将类型参数看作类型说明符,就像内置类型或者类类型说明符一样使用。我们类型参数前必须要使用class或者typename。

template<typename T1,class T2>
int compare(const T1 a, const T2 b);

在模板参数列表中,这两个关键字的含义相同,可以互换使用,一个模板参数列表中可以同时使用这两个关键字。

4.非类型模板参数

除了定义类型参数,还可以在模板中定义非类型参数,一个非类型参数表示一个值而非一个类型。我们可以通过一个特定的类型名而非关键字class或者typename来指定非类型参数。
当一个模板被实例化时,非类型参数被第一个用户提供的或编译器推断出来的值所替代。这些值必须是常量表达式,从而允许编译器在编译时实例化模板。

template<unsigned N, unsigned M>
int comper(const char(&arr1)[N], const char(&arr2)[M])
{
	return strcmp(arr1, arr2);
}
int main()
{
	//实例化出int comper(const char(&arr1)[6], const char(&arr2)[5])
	comper("hello", "word");
	return 0;
}

非类型模板参数的模板实参必须是常量表达式。

3.类模板

类模板是用来生成类的蓝图的。与函数模板不同的是,编译器不能为类模板推断模板类型参数。为了使用类模板我们必须在模板名后的尖括号中提供额外信息。

1.定义模板

类模板的定义格式:

template<class T1, class T2, ..., class Tn>
class 类模板名
{
	// 类内成员定义
};

例如:

template<class T1, class T2>
class A
{
public:
	A(){}
	A(T1 a, T2 b)
	{
		_a = a;
		_b = b;
	}
	void Geta()
	{
		cout << _a << endl;
	}
	void Getb()
	{
		cout << _b << endl;
	}
	~A();
private:
	T1 _a;
	T2 _b;
};
//类模板中函数放在类外进行定义时,需要加模板参数列表
template<class T1, class T2>
A<T1, T2>::~A()
{
	_a = 0;
	_b = 0;
}
int main()
{
	A<int, char> a(10,'a');
	a.Geta();
	a.Getb();
	return 0;
}

注意:类模板中函数放在类外进行定义时,需要加模板参数列表。

2.实例化模板

我们已经看到,当我们使用一个类模板时,我们必须提供额外的信息。我们现在知道这些额外信息是显示模板实参列表,它们被绑定到模板参数,编译器使用这些模板实参来实例化出特定的类。
例如上面代码中

class A
{
public:
	A() {}
	A(int a, char b)
	{
		_a = a;
		_b = b;
	}
	void Geta()
	{
		cout << _a << endl;
	}
	void Getb()
	{
		cout << _b << endl;
	}
	~A()
	{
		_a = 0;
		_b = 0;
	}
private:
	int _a;
	char _b;
};

一个类模板的每个实例都会形成一个独立的类。

3.模板的成员函数

与其他任何类相同,我们既可以在类模板内部定义成员函数,也可以在类模板外部定义成员函数,且定义在类模板内的成员函数被隐式声明为内敛函数。我们在上面已经见到了如何在类模板内部和类模板外部定义成员函数了。

总结

C++的内存申请与释放是我们后面的基础,模板则为我们了解容器有很深的作用,所以这两个需要我们重点学习。

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

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

相关文章

蓝桥杯每日一题2023.9.27

4408. 李白打酒加强版 - AcWing题库 题目描述 题目分析 对于这题我们发现有三个变量&#xff0c;店&#xff0c;花&#xff0c;酒的数量&#xff0c;对于这种范围我们使用DP来进行分析。 dp[i][j][k]我们表示有i个店&#xff0c;j朵花&#xff0c;k单位酒的集合&#xff0c…

逆向入门及实战

一、逆向工程介绍 1.1 什么是逆向工程 提到逆向工程可能大多数人第一印象就是非道德层面的软件破解&#xff0c;其实不然&#xff0c;逆向工程又称为逆向技术&#xff0c;是一种产品设计技术再现过程&#xff0c;即对一项目产品进行逆向分析及研究&#xff0c;从而演绎并得出该…

git 过滤不需要提交的目录和文件

项目根目录下&#xff08;.git同级目录&#xff09;添加.gitignore文件 .DS_Store .idea npm-debug.log yarn-error.log /node_modules /log/**.log /config.js

【2023年11月第四版教材】第15章《风险管理》(合集篇)

第15章《风险管理》&#xff08;合集篇&#xff09; 1 章节说明2 管理基础2.1 风险的属性2.2 风险的分类★★★2.3 风险成本★★★2.4 管理新实践 3 管理过程4 管理ITTO汇总★★★5 过程1-规划风险管理6 过程2-识别风险6.1 识别风险★★★6.2 数据收集★★★6.3 数据分析★★★…

第一次作业题解

第一次作业题解 P5717 【深基3.习8】三角形分类 思路 考的是if()的使用,还要给三条边判断大小 判断优先级&#xff1a; 三角形&#xff1f;直角、钝角、锐角等腰等边 判断按题给顺序来 代码 #include <stdio.h> int main() {int a 0, b 0, c 0, x 0, y 0, z 0…

使用vpn/代理后电脑无法正常上网

有时候当我们关闭VPN后&#xff0c;却发现不能正常连接到互联网了。 解决步骤&#xff1a; 办法一&#xff1a; 1. 找到右下角wifi图标&#xff0c;鼠标右键点击然后点击网络和Internet 设置 2. 进入控制面板选择代理 3. 将自动检测打开&#xff0c;把使用代理服务器关闭 办法…

【C++入门指南】类和对象(上)

【C杂货店】类和对象&#xff08;上&#xff09; 一、面向过程和面向对象初步认识二、类的引入三、类的定义四、类的访问限定符及封装4.1 访问限定符4.2 封装 五、类的作用域六、类的实例化七、类对象模型7.1 类对象的存储规则7.2 例题7.3结构体内存对齐规则 八、this指针8.2 t…

【Java 进阶篇】使用 SQL 进行排序查询

在数据库中&#xff0c;我们经常需要对查询的结果进行排序&#xff0c;以便更容易地理解和分析数据。SQL&#xff08;Structured Query Language&#xff09;提供了强大的排序功能&#xff0c;允许我们按照指定的列对数据进行升序或降序排序。本文将详细介绍如何使用 SQL 进行排…

windows系统删除网络适配器

此电脑&#xff0c;右键&#xff0c;管理 打开本机设备管理器 其中找到网络适配器&#xff1a; 选中要删除的&#xff0c;右键点击“卸载设备”&#xff0c;点击卸载即可完成。

玩转Mysql系列 - 第24篇:如何正确的使用索引?

这是Mysql系列第24篇。 学习索引&#xff0c;主要是写出更快的sql&#xff0c;当我们写sql的时候&#xff0c;需要明确的知道sql为什么会走索引&#xff1f;为什么有些sql不走索引&#xff1f;sql会走那些索引&#xff0c;为什么会这么走&#xff1f;我们需要了解其原理&#…

GEO生信数据挖掘(三)芯片探针ID与基因名映射处理

检索到目标数据集后&#xff0c;开始数据挖掘&#xff0c;本文以阿尔兹海默症数据集GSE1297为例 目录 处理一个探针对应多个基因 1.删除该行 2.保留分割符号前面的第一个基因 处理多个探针对应一个基因 详细代码案例一删除法 详细代码案例二 多个基因名时保留第一个基因名…

前端JavaScript入门到精通,javascript核心进阶ES6语法、API、js高级等基础知识和实战 —— Web APIs(二)

思维导图 一、事件监听&#xff08;绑定&#xff09; 1.1 事件监听 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta http-equiv"X-UA-Compatible" content"IEedge"><meta name&q…

VC++判断程序是否已经运行;仅运行一次

VC判断程序是否已经运行&#xff1b;仅运行一次 BOOL CClientApp::InitInstance() {...//判断程序是否已经运行&#xff1b;仅运行一次CreateMutex(NULL,true,_T("xxxxx")); //xxxxx&#xff1a;为程序标识码if(GetLastError()ERROR_ALREADY_EXISTS) { AfxMess…

JS对象数组去重

JS对象数组去重 一、数组去重1.使用 new Set()2.使用 indexOf 去重3.使用 includes 去重4.使用 hasOwnProperty5.使用 filter6.使用递归7.利用 Map 数据结构去重8.使用用 reduce includes9.使用 new Set() 的简化 二、对象数组去重1.使用 new Map() 和 filter2.使用reduce3.使…

华为云HECS云服务器docker环境下安装nginx

前提&#xff1a;有一台华为云服务器。 华为云HECS云服务器&#xff0c;安装docker环境&#xff0c;查看如下文章。 华为云HECS安装docker-CSDN博客 一、拉取镜像 下载最新版Nginx镜像 (其实此命令就等同于 : docker pull nginx:latest ) docker pull nginx查看镜像 dock…

GB/T 14710-2009 医用电器环境要求及试验方法

举个例子&#xff1a; 应符合GB/T 14710-2009中气候环境试验II组&#xff0c;机械环境试验II组的要求。 气候环境试验II组&#xff0c;机械环境试验II组&#xff1f; 这是2个属性&#xff0c;先按特定的条件分组&#xff0c;分组后&#xff0c;应该满足该组的特定要求。这个标…

A+CLUB管理人支持计划第九期 | 仟富来资产

免责声明 本文内容仅对合格投资者开放&#xff01; 私募基金的合格投资者是指具备相应风险识别能力和风险承担能力&#xff0c;投资于单只私募基金的金额不低于100 万元且符合下列相关标准的单位和个人&#xff1a; &#xff08;一&#xff09;净资产不低于1000 万元的单位&…

为什么 SetWindowsHookEx 采用 HINSTANCE 参数?

有开发者问了这样一个问题&#xff1a;既然 SetWindowsHookEx 的第一个参数总是会被转换为一个文件名&#xff0c;那为什么它的传参类型是 HINSTANCE 呢&#xff1f;这岂不是多此一举&#xff1f; 原因是这样的&#xff1a;在 16 位 Windows 系统上&#xff0c;它不是这样工作…

MATLAB 与 Cruise 的联合仿真

文章目录 检查matlab是否安装了编译器在 MATLAB 中添加路径联合仿真示例 检查matlab是否安装了编译器 第一步&#xff0c;先检查matlab是否安装了编译器&#xff1a; 关于编译器的配置&#xff0c;可以查看&#xff1a; https://blog.csdn.net/chengkai730/article/details/1…

FPGA的数字钟带校时闹钟报时功能VHDL

名称&#xff1a;基于FPGA的数字钟具有校时闹钟报时功能 软件&#xff1a;Quartus 语言&#xff1a;VHDL 要求&#xff1a; 1、计时功能:这是数字钟设计的基本功能&#xff0c;每秒钟更新一次,并且能在显示屏上显示当前的时间。 2、闹钟功能:如果当前的时间与闹钟设置的时…