C++动态内存区域划分、new、delete关键字、泛型编程、函数模版、类模版

目录

一、C/C++中程序的内存区域划分

为什么会存在内存区域划分?

二、new关键字

1、内置类型的new/delete使用方法:

2、new和delete的本质

3、常见面试题——malloc/free和new/delete的区别

三、模版

1、泛型编程

2、函数模版

(1)、引言

(2)、概念

3、模版原理

4、模版的实例化

5、类模版

6、匹配调用规则:

7、类模板注意事项:


一、C/C++中程序的内存区域划分

为什么会存在内存区域划分?

因为不同数据有不同的存储需求,各区域满足不同的需求。

例如:

(1)、一些临时变量,局部变量存储在栈区

(2)、常用数据结构或一些算法(如归并排序)中会用到动态内存开辟,该内存是在堆区申请。

(3)、全局变量、静态变量存储在数据段(静态区)。

(4)、一些只读数据,如字符常量、可执行代码(指令)存储在代码段(常量区)。

二、new关键字

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

1、内置类型的new/delete使用方法:

	//动态申请一个int型
	int* a = new int;

	//动态申请一个int型,并初始化为10
	int* a1 = new int(10);

	//动态申请一个int型数组
	int* arr1 = new int[10];
	
	//动态申请一个int型数组,并初始化
	int* arr2 = new int[3] {1, 2, 3};

	//若为完全初始化,则后面的值会默认初始化为0
	int* arr3 = new int[10] {1, 2, 3};

	//每new一个变量,用完后一定要delete
	delete a;
	delete a1;
	//注意数组的区别
	delete[] arr1;
	delete[] arr2;
	delete[] arr3;

注意:使用new未初始化时,不会有默认值,而是随机值。

2、new和delete的本质

new的本质是为了解决动态申请的自定义类型对象的初始化问题。

因为C语言中malloc函数不能给自定义类型进行初始化:

	//malloc不能给自定义类型初始化
	A* p1 = (A*)malloc(sizeof(A));

所以这时就可以用new和delete:

(1)、new的本质:开空间+调用构造函数初始化;

(2)、delete的本质:先调用析构函数+释放空间。

如下:

	//new给自定义类型初始化
	//调用默认构造函数
	A* p2 = new A();
	//调用带参数构造函数
	A* p3 = new A(1);

	//给对象数组初始化
	//(1)、有名对象
	A aa1, aa2, aa3;
	A* arr1 = new A[3]{ aa1,aa2,aa3 };
	//(2)、匿名对象
	A* arr2 = new A[3]{ A(),A(),A()};
	//(3)、隐式类型转换
	A* arr3 = new A[3]{ 1,2,3 };

注意:

(1)、内置类型的底下申请释放,new和malloc除了用法上,没有区别;

(2)、malloc申请失败会返回NULL,而new申请失败会抛异常;

3、常见面试题——malloc/free和new/delete的区别

从用法+原理角度区分:

malloc/freenew/delete
共同点是:都是从堆上申请空间,并且需要用户手动释放。
不同点是:
1. mallocfree是函数,newdelete是操作符
2. malloc申请的空间不会初始化,new可以初始化
3. malloc申请空间时,需要手动计算空间大小并传递,new只需在其后跟上空间的类型即可, 如果是多个对象,[]中指定对象个数即可
4. malloc的返回值为void*, 在使用时必须强转,new不需要,因为new后跟的是空间的类型
5. malloc申请空间失败时,返回的是NULL,因此使用时必须判空,new不需要,但是new 要捕获异常
6. 申请自定义类型对象时,malloc/free只会开辟空间,不会调用构造函数与析构函数,而new 在申请空间后会调用构造函数完成对象的初始化,delete在释放空间前会调用析构函数完成 空间中资源的清理。

三、模版

1、泛型编程

泛型编程通俗来讲就是写跟具体类型无关的代码,广泛的编程。是代码复用的一种手段。模板是泛型编程的基础。

2、函数模版

(1)、引言

我们以交换函数为例,看下列代码:

void Swap(int& p1, int& p2)
{
	int tmp = p1;
	p1 = p2;
	p2 = p1;
}

void Swap(char& p1, char& p2)
{
	char tmp = p1;
	p1 = p2;
	p2 = p1;
}

void Swap(double& p1, double& p2)
{
	double tmp = p1;
	p1 = p2;
	p2 = p1;
}

这是用函数重载的知识,但函数重载也有不好的地方:

1. 重载的函数仅仅是类型不同,代码复用率比较低,只要有新类型出现时,就需要用户自己增加对应的函数。
2. 代码的可维护性比较低,一个出错可能所有的重载均出错。
正如上述各函数,我们发现好像只有类型不同,这时我们就可以用模版来减少代码的冗余。
(2)、概念
函数模板代表了一个函数家族,该函数模板与类型无关,在使用时被参数化,根据实参类型产生函数的特定类型版本。
格式:
template<typename T1, typename T2,......,typename Tn>
返回值类型 函数名(参数列表){}
例如上述代码,我们可以写成:
template<typename T>

void Swap(T& p1, T& p2)
{
	T tmp = p1;
	p1 = p2;
	p2 = tmp;
}

当有多种类型时,就可以写多个typename(或class):

注意:

typename是用来定义模板参数关键字,也可以使用class(切记:不能使用struct代替class)

3、模版原理

注意一点:
模版处只有一个函数,为什么会是调用的两个函数?这就是模版的原理所造成的现象:
原理:
函数模板是一个蓝图,它本身并不是函数,是编译器用使用方式产生特定具体类型函数的模具。所以其实模板就是将本来应该我们做的重复的事情交给了编译器。
在编译器编译阶段,对于模板函数的使用,编译器需要根据传入的实参类型来推演生成对应类型的函数以供 调用。比如:当用double类型使用函数模板时,编译器通过对实参类型的推演,将T确定为double类型,然 后产生一份专门处理double类型的代码,对于字符类型也是如此。

4、模版的实例化

用不同类型的参数使用函数模板时,称为函数模板的实例化。模板参数实例化分为:隐式实例化和显式实例化。
模版参数语法很类似函数参数,只是函数参数定义的是形参对象,而模版定义的是类型。

(1)、推演实例化:例如上述内容,是编译器自己识别模版是什么类型,就叫推演实例化。

(2)、显示实例化:

//单类型模版显示调用
Swap<int>(1, 1);
//多类型模版显示调用
Test<int, double>(1, 5.5);

有一些情况只能用显示实例化,比如无参函数,传不了参数,就不能靠推演确定类型,这时只能显示实例化模版:

5、类模版

(1)、类模版定义格式

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

特别注意:类模版必须显示实例化:

然后模版与类之间不能存在函数使用模版,同理想要函数模版,中间也不能穿插类使用模版。

6、匹配调用规则:

1、有现成的函数可以匹配就调用现成。

2、有合适的就调用合适的,没有就将就。

7、类模板注意事项:

(1)、使用类模版后,当成员函数的声明与定义分开写时,会有改变:

template<class T>
class Test
{
public:
	//构造函数
	//声明
	Test(T i);
private:
	int _i;
};

//定义
template<class T>
Test<T>::Test(T i)
{
	T tmp;
}

(2)、特别注意:

普通类:类名就是类型。

类模板实例化的类:类名不是类型,类名<数据类型>才是整个类的类型。

也就是说:显示实例化类模板的类型不同,它们就是不同的类:

  • 18
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 13
    评论
评论 13
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

成工小白

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值