【C++】是内存管理,但C++ !! && 模板初阶

2023-11-05

目录

一,回望C语言内存

二, C++  内存管理方式 

1. 内置类型

2. 自定义类型

3. new & malloc 返回内容区别

4. operator new   & operator  delete 

5. malloc/free和new/delete的区别总结

6. 定位new表达式(placement-new) (了解)

三,模板初阶

1.  泛型编程——概念

2. 函数模板

(1. 模板实例化

   概念 

(2. 显式实例化

(3. 隐式实例化

3.  类模板——【练习】实现栈的Push


一,回望C语言内存

尝试给出下面答案:

结果:

解析:

二, C++  内存管理方式 

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

1. 内置类型

 对于内置类型 new / delete 与 malloc & free 没有本质的区别

void Test()
{   // 申请一个int空间
	int* z1 = new int;
    // 申请5个int的数组
	int* z2 = new int[5];
	// 申请一个int空间,初始化为5
	int* z3 = new int(5);

    // 销毁
	delete z1;
	delete[] z2; // 销毁类型要匹配
	delete z3;
	// 对于内置类型 new / delete 与 malloc & free 没有本质的区别
	// 仅仅new只是用法上简化了
}

注意: C++里面没有支持 C语言 realloc 的创新,C++接口不支持扩容。 

2. 自定义类型

new 的原理
  1. 调用 operator new 函数申请空间
  2. 在申请的空间上执行构造函数,完成对象的构造
delete 的原理
  1. 在空间上执行析构函数,完成对象中资源的清理工作
  2. 调用 operator delete 函数释放对象的空间
new T[N] 的原理
  1. 调用 operator new[] 函数,在 operator new[] 中实际调用 operator new 函数完成 N 个对
象空间的申请
  2. 在申请的空间上执行 N 次构造函数
delete[] 的原理
  1. 在释放的对象空间上执行 N 次析构函数,完成 N 个对象中资源的清理
  2. 调用 operator delete[] 释放空间,实际在 operator delete[] 中调用 operator delete 来释
放空间

struct  B
{
public:
	B(int b)
	{
		k = b;
		cout << "B" << endl;
	}

	~B()
	{
		cout << "~B" << endl;
	}
private:
	int k;
};

struct  A
{
public:
	A(int sum, int b)
		: _sum(sum)
		, _b(b)
	{
		cout << "A" << endl;
	}

	~A()
	{
		cout << "~A" << endl;
	}
private:
	int _sum;
	B _b;
};

int func()
{
	A* p1 = new A(10, 20); // 1.内置类型初始化  2. 自定义类型调用其构造函数
	A* p2 = new A[10];     
	A* p3 = new A[10]{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}; // 多实例,并且初始化
	delete p1;             // 1. 销毁内置类型   2. 调用析构函数
	delete[] p2;           // 销毁类型一定要匹配使用, 否则可能出现报错
	delete[] p3;
	return 0;  // 返回码
}

3. new & malloc 返回内容区别

(1) malloc 失败 返回 null 指针

(2) new  失败    会返回异常 (异常后面再着重开始讲)

void test()
{
	try   // 其中如果new失败会进入异常,也就是catch
	{
		// 当内存申请超过一定限度,会申请失败
	// malloc 失败会返回null
		char* p1 = (char*)malloc(sizeof(char) * 1024u * 1024u * 1024u * 1024u * 1024 * 1024);
		printf("%p\n", p1);
		// new  失败会抛异常
		char* p2 = new char[1024 * 1024 * 1024 * 1024 * 1024 * 1024];
		printf("%p\n", p2);

		free(p1);
		delete[] p2;
	}
	catch (const std::exception& e)   // 打印最近的异常信息
	{
		cout << e.what() << endl;
	}
}

注意: 面对new 失败返回异常 会用一个try 来包含 需要new 的代码。

4. operator new   & operator  delete 

       new和delete是用户进行 动态内存申请和释放的操作符operator new 和operator delete是系统提供的 全局函数new在底层调用operator new全局函数来申请空间, delete在底层通过 operator delete全局函数来释放空间。
(总得来说 operator new & delete,并不是函数重载,而是帮助new,delete实现其机制的全局函数)

 

operator new该函数实际通过malloc来申请空间,当malloc申请空间成功时直接返回;申请空间失败,尝试执行空间不足应对措施,如果改应对措施用户设置了,则继续申请否则抛异常。( new -> operator new ->  malloc  ->  指针 或  异常

operator delete: 该函数最终是通过free来释放空间的。 

5. malloc/freenew/delete的区别总结

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

6. 定位new表达式(placement-new) (了解)

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

如: new  (p1)  A (10)  // 其中 p1 通过malloc已开辟空间但未初始化,A是类型 , 10是调用自定义构造函数初始化的值。

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

struct  B
{
public:
	B(int b)
	{
		k = b;
		cout << "B" << endl;
	}

	~B()
	{
		cout << "~B" << endl;
	}
private:
	int k;
};

void test1()
{
	B* p3 = (B*)malloc(sizeof(B));
	// malloc 的空间并未初始化
	new (p3) B(100); // 给构造函数参数 100
	p3->~B();
	delete p3;
}


三,模板初阶

1.  泛型编程——概念

函数模板代表了一个函数家族,该函数模板与类型无关,在使用时被参数化,根据实参类型产生函数的特定 类型版本。

我们肯定写过这样的编程

void Swap(int& z1, int& z2)
{
	int tmp = z2;
	z2 = z1;
	z1 = tmp;
}

交换函数,一般只能处理一种数据; 而一个程序需要交换多种类型数据,这样我们需要创建多种类型交换函数,这个过程是重复,低效的

泛型编程:编写与类型无关的通用代码,是代码复用的一种手段。模板是泛型编程的基础。

2. 函数模板

原理:函数模板是一个蓝图,它本身并不是函数,是编译器用使用方式产生特定具体类型函数的模具。所以其实模 板就是将本来应该我们做的重复的事情交给了编译器。

编译器使用模板步骤:

1. 推演 类型

2. 模板实例化

下面简单展示一下:

template <class T>   // 或者 <template T> T只是取名字
void Swap(T& z1, T& z2)
{
	T tmp = z2;
	z2 = z1;
	z1 = tmp;
}

int main()
{
	int i = 1, b = 2;
	Swap(i, b);
	return 0;
}

在编译器编译阶段,对于模板函数的使用,编译器需要根据传入的实参类型来推演生成对应类型的函数以供调用。比如:当用double类型使用函数模板时,编译器通过对实参类型的推演,将T确定为double类型,然后产生一份专门处理double类型的代码,同理这就是我们交换函数的模板, 中间编译器会进行推导类型,编译更复杂,时间会稍久一些。

注意:交换用的是同一套模板,但调用的函数并不是同一个函数。(编译器自动帮我们创建不同类型的交换函数)

关键的来了: 以后不需要自己创建交换函数C++库中有一套交换函数,在std 命名空间里面。

操作如下:

#include <iostream>
using namespace std;
int main()
{
	int i = 1, b = 2;
	double z = 1.1, k = 2.2;
	swap(i, b);   // 小写就行
	swap(z, k);   // 模板推导,需要类型相同,否则报推导类型不确定
	return 0;
}

(1. 模板实例化

概念 

用不同类型的参数使用函数模板时,称为函数模板的 实例化。  模板参数实例化分为 隐式实例化    和    显式实例化

(2. 显式实例化

在函数名后的<>中   指定模板参数   的实际类型

template <class T>   // 或者 <template T>
T* func1(int n)
{
	T* k1 = new T(n);
}


int main(void)
{
 int a = 10;
 double b = 20.0;
 
 // 显式实例化
 Add<int>(a, b);  1. 普通
 func1<A>(10);    2. 特殊,必须要显式否则怎么推演?计算机:二进制给你,你来推演!
 return 0;
}

(3. 隐式实例化

让编译器根据实参   推演模板 参数的实际类型

#include <iostream>
using namespace std;

int ADD(size_t & z1, size_t & z2)  // size_t是标准C库中定义的,它是一个基本的与机器相关的无符号整数的C/C + +类型
  // 就简单理解为 unsiged int 主要防止 负数
{
	return z1 + z2;
}

template <class T>   // 或者 <template T>
T ADD(const T& z1, const T& z2)
{
	return z1 + z2;
}

template <class B1, class B2> // 可以提供多个模板
B1 ADD(B1& b1, B2& b2)
{
	return b1;
}

int main()
{
	// 隐式实例 ——通过编译器自动推导, 不人为干预。
	int i = 1, b = 2;
	double z = 1.1, k = 2.2;
	cout << ADD(i, (int)z) << endl; // 模板 不支持1.1 -> 1 隐式转化,但可以提前,进入的是
	cout << ADD(1.2, (double)1) << endl;

	// 思考;调用哪一个函数
	cout << ADD(i, b) << endl;  // 计算机会调用现成的函数,
	// 调用 int ADD(int& z1, int& z2) ; 模板实例化需要通过编译器,所以优先级比较低。
	cout << ADD(z, i) << endl;  // 两种类型,则会选择多类型模型 B1 ADD(B1& b1, B2& b2)
	return 0;
}

3.  类模板——【练习】实现栈的Push

#include <iostream>
using namespace std;
#include <assert.h>

template <class T>
class stack
{
public:
	stack(int capacity = 4)
		: _size(0)
		, _capacity(capacity)
		, plist(nullptr)
	{
		plist = new T[_capacity];
	}

	void Push(T x)
	{
		assert(plist);
		if (_size == _capacity)
		{
			// 1. 开空间
			// 2. 拷贝旧数据
			T* tmp = new T[_capacity * 2];
			if (plist)   // plist 首先是空指针,如果是是第一次不需要拷贝旧数据
			{
				memcpy(tmp, plist, sizeof(T) * _capacity);
				// 如果申请成功
				_capacity *= 2;
				delete[] plist;
				plist = tmp;
			}
		}
		plist[_size++] = x;
		cout << plist[_size - 1] << endl;
	}

	void Pop()
	{
		assert(_size > 0)
		_size--;
	}

	const T& TopStack()  // 返回栈顶元素,用的是引用,可以修改 私密成员,所以需要const修饰
	{
		assert(_size > 0);  // 防止栈为空
		return plist[_size - 1];
	}

	~stack()
	{
		delete[] plist;
		_size = _capacity = 0;
	}
private:
	T* plist;
	int _size;
	int _capacity;
};

void func()
{
	try
	{
		stack<int> p1;  // 其中如果new失败会进入异常,也就是catch
		stack<char> p2;
		p1.Push(1);
		p1.Push(2);
		p1.Push(3);
		p1.Push(4);
		p1.Push(5);

		cout << p1.TopStack() << endl;
	}
	catch (const std::exception& e)
	{
		cout << e.what() << endl; // 打印最近的异常信息
	}
}

总结:

  • 类模板定义与声明不能分离(同文件允许分离),否则会报错。
  • 一般类实例化时,不带类型,所以一般类模板都要使用显示实例化。
  • 在C++中,类模板在头文件中定义,因此可以写成“ .hpp”的头文件,意思是不仅有声明而且还有实现,一般是类模板实现。

结语

本小节就到这里了,感谢小伙伴的浏览,如果有什么建议,欢迎在评论区评论;如果给小伙伴带来一些收获请留下你的小赞,你的点赞和关注将会成为博主创作的动力。

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

【C++】是内存管理,但C++ !! && 模板初阶 的相关文章

  • 使用内部构造函数实例化类

    我有一个类 其构造函数被定义为内部 这意味着我无法实例化它 虽然这可能有道理 但出于调试和研究目的 我仍然愿意做一次 是否可以通过反射来做到这一点 我知道我可以访问私有 内部成员 但是我可以调用内部构造函数吗 或者 由于构造函数没有做任何重
  • boost::interprocess 准备好迎接黄金时间了吗? [关闭]

    很难说出这里问的是什么 这个问题是含糊的 模糊的 不完整的 过于宽泛的或修辞性的 无法以目前的形式得到合理的回答 如需帮助澄清此问题以便重新打开 访问帮助中心 help reopen questions 我正在开发一个由内存映射文件支持的线
  • 何时对向量进行归一化?

    我正在学习 XNA 并且在几乎所有的教育套件中都可以找到http creators xna com en US http creators xna com en US 我总是看到向量上对 Normalize 的调用 我知道归一化基本上将向量
  • C/C++ 中随机数生成器的实现[重复]

    这个问题在这里已经有答案了 我对 C 中随机数生成器的实现有点困惑 它也与 C 中的明显不同 如果我理解正确 对 srand seed 的调用会以某种方式初始化可通过 rand 访问的隐藏变量 种子 该变量又将函数指向预先生成的序列 例如例
  • 在 MVC 类上创建主键字段

    我是 MVC 和 C 新手 我只是偶然发现它并发现它很有趣 我遇到了一个不允许我继续的问题 这是我的代码 using System using System Collections Generic using System Linq usi
  • 为什么 LinkedList 通常比 List 慢?

    我开始在我的一些 C 算法中使用一些 LinkedList 而不是列表 希望能够加快速度 然而 我注意到他们只是感觉更慢 像任何优秀的开发人员一样 我认为我应该尽职调查并验证我的感受 所以我决定对一些简单的循环进行基准测试 我认为用一些随机
  • initializer_list 和默认构造函数重载决策

    include
  • 首先EntityFramework数据库 - 类型映射 - 将binary(8)从SQL映射到C#中的int

    在 SQL 内部 我有一个主键为二进制 8 的表 当我使用该表添加到我的模型中时Update Model from Database我可以看到该列有 type Binary 在 C 中 我将该列设为byte 我可以将该列映射到 int 吗
  • 使用正则表达式匹配以“Id”结尾的单词?

    如何组合一个正则表达式来匹配以 Id 结尾的单词并进行区分大小写的匹配 试试这个正则表达式 w Id b w 允许前面的单词字符Id和 b确保Id位于单词末尾 b是字边界断言
  • IClaimsTransformation 未触发

    我尝试过实施一个IClaimsTransformation我在 ASP NET CORE 3 1 Web 应用程序中找到的类 public class ClaimsTransformer IClaimsTransformation publ
  • 基于 C++ 范围的 for 循环

    尝试使用基于范围的 for 循环执行某些操作 可以使用常规的 for 循环来完成 如下所示 vector
  • 为什么我的 ITexthandler 不工作?我正在尝试将 XML 解析为 ITextSharp 文档

    我正在使用 Visual Developer 2010 MVC 3 c 我正在尝试将 XML 解析为 iTextSharp 文档 如下所示 ITextHandler textHandler new ITextHandler doc text
  • fscanf 和 EOF 中的否定扫描集

    我的文件中有一个以逗号分隔的字符串列表 姓名 1 姓名 2 姓名 3 我想跳过所有逗号来阅读这些名字 我写了以下循环 while true if fscanf file my string 1 break 然而 它总是比预期多执行一次 给定
  • C# - 命名空间内的类型声明

    在命名空间内而不是在类中声明类型的可能用途是什么 For ex namespace Test public delegate void Ispossible 这是有效的并且不会产生任何编译错误 但我无法想象为什么我们会以这种方式声明它而不是
  • 如何在Linux上构建GLFW3项目?

    我已经使用 cmake 和 make 编译了 glfw3 和包含的示例 没有出现任何问题 开始编写我的第一个项目 作为 opengl 和 glfw 的新手 并且对 C 和 CMake 没有经验 我正在努力理解示例构建文件 甚至要链接哪些库和
  • 统一;随机物体移动[关闭]

    Closed 这个问题需要多问focused help closed questions 目前不接受答案 我正在制作一款机器人战斗游戏 我希望敌人随机移动 然后有时会向敌人移动 我希望运动包含在其中的代码 else if avoid fal
  • Visual Studio 2015默认附加库

    当我在 VS 2015 中创建一个空项目时 它会自动将这些库放入 附加依赖项 中 kernel32 lib user32 lib gdi32 lib winspool lib comdlg32 lib advapi32 lib shell3
  • Crypto++ 和压缩 EC 密钥

    如何在 Crypto 中生成压缩的 ECDSA 密钥 AutoSeededRandomPool prng ECDSA
  • 如何设置 Swashbuckle 与 Microsoft.AspNetCore.Mvc.Versioning

    我们有asp net core webapi 我们添加了Microsoft AspNetCore Mvc Versioning and Swashbuckle拥有招摇的用户界面 我们将控制器指定为 ApiVersion 1 0 Route
  • 如何获取通过网络驱动器访问的文件的 UNC 路径?

    我正在 VC 中开发一个应用程序 其中网络驱动器用于访问文件 驱动器由用户手动分配 然后在应用程序中选择驱动器 这会导致驱动器并不总是映射到相同的服务器 我该如何获取此类文件的 UNC 路径 这主要是为了识别目的 这是我用来将普通路径转换为

随机推荐

  • 【JavaScript高级】class类、ES6实现继承、ES6对象增强

    文章目录 class类 构造函数 访问器方法 getter和setter 静态方法 ES6实现继承 extends super 继承内置类 类的混入 了解 ES6对象的增强 字面量的增强 解构 解构相关应用 多态 参考 class类 ES6
  • java毕业论文云笔记_java毕业设计_springboot框架的云笔记记事本

    这是一个基于java的毕业设计项目 毕设课题为springboot框架的云笔记记事本 是一个采用b s结构的javaweb项目 开发工具eclipsei eclipse 项目框架jsp springboot mybatis 云笔记记事本采用
  • 命令式编程和声明式编程

    一 命令式编程 命令 机器 如何去做事情 how 这样不管你想要的是什么 what 它都会按照你的命令实现 注重过程 用详细的命令机器怎么去处理一件事情以达到你想要的结果 例如你想通过点击改变页面中某一个元素 首先要获取按钮 再给按钮添加点
  • 重新配对_Apple Watch配对失败的解决办法

    昨晚上我把那支用了将近三年的3代苹果表重置之后 打算重新配对 结果第一次碰上了配对失败的问题 弹窗提示我可能连接了不受信任的网络 前后用无线网和4G网络试了很多次 都是一样的结果 百度解决方法也完全没头绪 今天下午终于在苹果客服的帮助下解决
  • 这个落泪的男人叫王坚

    这个落泪的男人叫王坚 前段时间新闻报道 说一个名叫Watson的人工智能 花十几分钟读完2000万页医疗文献之后 解决了医生都束手无策的病情 听着感觉这人工智能跟打败世界棋手李世石的AlphaGO 阿尔法狗 相比弱爆了 但要知道那可是200
  • Vulkan入门(一)-环境配置.md

    文章目录 参考资料 简述 一 准备环境 1 1 开发环境 1 2 下载 SDK 1 3 安装SDK 1 4 安装驱动 1 5 运行示例程序 二 GLFW 安装 三 GLM 安装 四 手动编译示例代码 4 1 在编译示例代码的时候老是报错 找
  • MySQL 下载安装教程

    MySQL Community 8 0 安装教程 说明 步骤 说明 本教程只是 Windows 下 MySQL 的一种集成 IDE 的安装教程 安装此 IDE 还免去了手动在 Windows 上配置 MySQL 的麻烦 不过如果读者对 My
  • 国产 CAE 软件研发

    1 简介 国产 CAE 软件研发的特点 国家重大需求 自主知识产权 架构灵活 二次开发 通用化 定制化 2 CAE 软件研发 现状和意义 我国自主开发商业化程度高 通用性强的 CAE 软件 尤其是开发带良好图形用户界面的前后处理器的工作还十
  • RISC-V from scratch 5:机器模式

    RISC V from scratch 5 机器模式 接上一篇博客 我今天继续写 RISC V from scratch 系列博客 原本我打算将该英文系列全部翻译成中文 但原作者貌似没有把这一系列完成就咕咕了 因此本文的内容是我自己实践的内
  • 企业级远程桌面,需要考虑哪些核心因素?

    随着经济的发展 现代企业的管理模式也在发生改变 2020年初 新冠疫情爆发 所有的企业都无法按时返工 各个企业都开始寻求新的办公模式 埃森哲2021最新报告中有一个数据显示 87 的高管人员认为 远程劳动力为劳动人才市场打开了新天地 且扩大
  • 《信号与系统》解读 前言:经典教材的选择

    1 教材选择 信号与系统 系统的教材很多 分国内与国外教材 专题以 信号与系统 奥本海姆第二版为基础与主线 结合LTE 5G移动通信的工程实践需要 有选择性的对理论内容进行解读 2 主要内容 信号与系统 是美国麻省理工学院 MIT 的经典教
  • Stata计算可操纵性应计利润——基于琼斯模型

    说明 数据 变量名称来源于国泰安数据库 具体名称可见国泰安数据库资产负债表 利润表 本代码仅供参考 代码实现 基本Jones模型 Jones 1991 提出了经典的Jones模型 从营业收入变动和固定资产水平衡量企业应计利润的变动 clon
  • Java中文与Base64互转(解决中文乱码的问题)

    最近线上出现一个问题 前后端交互时 某些情况下 会有中文乱码的问题 解决思路 1 在后端先将中文转为 Base64 后再传递到前端 此中文在前端不做显示处理 2 前端将参数再传递回后端时 后端解析 Base64 得到中文字符串 packag
  • 9个免费的矢量图网站

    寻找一些特别的 为众所不知的矢量图网站不是一件容易的事情 又要高质量 又要免费使用 尽管鱼和熊掌不能兼得 但是谁叫我们碰到了互联网时代呢 谁叫我们知道一句台词 一切皆有可能呢 这些免费的矢量图网站是我在互联网上搜索到的 经过权衡和对比 选择
  • 设计模式——装饰模式

    装饰模式 1 装饰模式动机及定义 1 1模式动机 买了新房 毛胚房 需要装修 对新房进行装修并没有改变房子用于居住的本质 但它让房子变的更漂亮 更加满足居家的需求 在软件设计中 我们也可以用类似的技术对原有对象 新房 的功能进行扩展 装修
  • 由阿里巴巴一道笔试题看Java静态代码块、静态函数、动态代码块、构造函数等的执行顺序

    一 阿里巴巴笔试题 public class Test public static int k 0 public static Test t1 new Test t1 public static Test t2 new Test t2 pu
  • 第19课:生活中的访问模式——一千个读者一千个哈姆雷特

    用程序来模拟生活 从剧情中思考访问模式 访问模式 访问模式的模型抽象 代码框架 类图 基于框架的实现 模型说明 设计要点 优缺点 访问模式的优点 访问模式的缺点 实战应用 应用场景 故事剧情 光阴似箭 转眼间作为
  • 梁乾东:4.16黄金走势上涨冲破天际,今日黄金原油在线建议附解套

    消息面解析 周四 4月15日 美国原油上涨 因美国原油库存下降支持全球需求复苏的希望 总体来看 美国原油库存下降提振了对需求改善的乐观情绪 美联储褐皮书中对美国经济的乐观预期利好油价 此外早间沙特拦截了胡塞武装发射的两枚导弹和四架无人机 中
  • 互联网日报

    今日看点 嫦娥五号探测器成功实施近月制动 顺利进入环月轨道 中国移动香港实现5G独立组网 助力建设世界级智慧城市 哈啰出行换电业务启用全新品牌 小哈换电 开启两轮换电新纪元 翰森制药钟慧娟成中国及全球白手起家女首富 财富达1350亿元 交通
  • 【C++】是内存管理,但C++ !! && 模板初阶

    目录 一 回望C语言内存 二 C 内存管理方式 1 内置类型 2 自定义类型 3 new malloc 返回内容区别 4 operator new operator delete 5 malloc free和new delete的区别总结