c++ - 空间申请和释放 new/delete


一、c/c++内存分布

求下面各个变量的位置

// c/c++内存分布

int globalVar = 1;
static int staticGlobalVar = 1;
void Test()
{
	static int staticVar = 1;
	int localVar = 1;
	int num1[10] = { 1, 2, 3, 4 };
	char char2[] = "abcd";
	const char* pChar3 = "abcd";
	int* ptr1 = (int*)malloc(sizeof(int) * 4);
	int* ptr2 = (int*)calloc(4, sizeof(int));
	int* ptr3 = (int*)realloc(ptr2, sizeof(int) * 4);
	free(ptr1);
	free(ptr3);
}
选项: A.栈  B.堆  C.数据段(静态区)  D.代码段(常量区)  

1、 globalVar在哪里?____ 2、staticGlobalVar在哪里?____ 3、staticVar在哪里?____ 4、localVar在哪里?____ 5、num1 在哪里?____ 6、char2在哪里?____ 7、*char2在哪里?___ 8、pChar3在哪里?____ 9、*pChar3在哪里?____ 10、ptr1在哪里?____ 11、*ptr1在哪里?____
答案:1-5 CCCAA 6-11AAADAB

![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/4ea571fe70e04481baf829bd12b580aa.png

说明:

  1. 栈又叫堆栈–非静态局部变量/函数参数/返回值等等,栈是向下增长的。
  2. 内存映射段是高效的I/O映射方式,用于装载一个共享的动态内存库。用户可使用系统接口 创建共享共享内存,做进程间通信。(Linux课程如果没学到这块,现在只需要了解一下)
  3. 堆用于程序运行时动态内存分配,堆是可以上增长的。
  4. 数据段–存储全局数据和静态数据。
  5. 代码段–可执行的代码/只读常量。

二、new/delete 的使用

关键字: new
作用:申请空间
关键字:delete
作用:释放空间

如:


//new 和 delete的使用
int main()
{
	//new 申请一个变量空间	
	int* p = new int;	//(类型指针) 指针名 = new 类型

	//delete 释放一个变量空间	// delete 释放的指针
	delete p;

	//new 申请一个数组空间	
	int* pp = new int[10];	//(类型指针) 指针名 = new 类型[(申请的数组大小)]

	//delete 释放一块数组的空间
	delete[]pp;	//delete [] 释放的指针

	return 0;
}

注意:
释放数组的时候 delete 一定要和 [ ] 配对使用,不然只会释放该指针指向的第一个位置,这样就会造成内存泄漏。

三、malloc/free 和 new/delete 的对比

1、mallocfree是库函数,而newdelete是运算符(类似sizeof 属于运算符而不属于库函数) 。

2、malloc使用比new更复杂
如:

int main()
{
	//malloc申请一个变量空间
	int* p = (int*)malloc(sizeof(int));

	//new申请一个变量空间
	int* pp = new int;

	return 0;
}

在这里插入图片描述

3、malloc 不能对申请的空间进行初始化,而 new
如:

int main()
{
	//对一个变量初始化
	int* p = new int(10);	//在后面加括号再加初始化的值

	//对数组进行初始化
	int* pp = new int[5] {1, 2, 3, 4, 5};	//在后面加花括号再加初始化的值

	cout << *p << endl;

	for (int i = 0; i < 5; i++)
	{
		cout << pp[i] << " ";
	}

	return 0;
}

在这里插入图片描述

4、对于自定义类型,new能调用其构造函数,delete能调用其析构函数,而malloc/free不行。

如:

class A
{
public:
	A()
	{
		cout << "A()" << endl;
	}
	~A()
	{
		cout << "~A()" << endl;
	}
};

int main()
{
	//用malloc申请
	A* a = (A*)malloc(sizeof(A));
	free(a);

	cout << "划分线" << endl;

	//用new申请一个自定义类型空间
	A* p = new A;

	//用new申请一个自定义类型数组
	A* pp = new A[10];

	//释放
	delete p;
	delete[]pp;

	return 0;
}

在这里插入图片描述

5、在申请空间失败时malloc会返回NULL,而new则会抛出异常。

四、new/delete 的实现原理

operator new 函数(底层还是使用malloc申请空间)是 new 的底层实现。
operator delete 函数(底层还是使用free释放空间)是 delete 的底层实现。
operator new[] 函数是 new 类型[ ]的实现。
operator delete[] 函数是 delete[] 的实现。

对于内置类型

如果申请的是内置类型的空间,newmallocdeletefree基本类似,不同的地方是:
new/delete申请和释放的是单个元素的空间,new[]delete[]申请的是连续空间,而且new在申
请空间失败时会抛异常,malloc会返回NULL

对于自定义类型

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来释空间。

练习1
使用 char* p = new char[100]申请一段内存,然后使用delete p释放,有什么问题?( )
A.会有内存泄露
B.不会有内存泄露,但不建议用
C.编译就会报错,必须使用delete []p
D.编译没问题,运行会直接崩溃
答案:B

A.对于内置类型,此时delete就相当于free,因此不会造成内存泄漏
B.正确
C.编译不会报错,建议针对数组释放使用delete[],如果是自定义类型,不使用方括号就会运行时错误
D.对于内置类型,程序不会崩溃,但不建议这样使用

练习2
ClassA *pclassa=new ClassA[5];delete pclassa;c++语言中,类ClassA的构造函数和析构函数的执行次数分别为( )
A.5,1
B.1,1
C.5,5
D.程序可能崩溃
答案:D

A.申请对象数组,会调用构造函数5次,delete由于没有使用[],此时只会调用一次析构函数,但往往会引发程序崩溃
B.构造函数会调用5次
C.析构函数此时只会调用1次,要想完整释放数组空间,需要使用[]
D.正确

五、匹配问题

关于malloc/free 和 new/delete 申请和释放匹配

1、对于内置类型,是可以的,但是不建议,最好还是配对使用

//内置类型
int main()
{	
	//new 申请
	int* p = new int;

	//free 释放
	free(p);

	//malloc 申请
	int* pp = (int*)malloc(sizeof(int));

	//delete 释放
	delete pp;

	return 0;
}

会报警告:
在这里插入图片描述
在这里插入图片描述

2、对于自定义类型来说很容易出现问题,如下面

(1)申请数组

class A
{
public:
	A():_a(10)
	{
		cout << "A()" << endl;
	}
	~A()
	{
		cout << "~A()" << endl;
	}
private:
	int _a;
};

int main()
{
	cout << sizeof(A) << endl;	//大小为 4
	
	// 用 malloc 申请 申请空间的大小为 80字节
	A* pp = (A*)malloc(sizeof(A) * 20);

	//用 new  申请  申请的空间大小为 88字节(原本是应为 80 字节,但是多出来的8字节为保存申请多少个A )
	A* p = new A[20];	

	//用 free 释放
	free(p);	//直接崩掉,因为p的地址为申请的88个字节中的第8个字节地址,所以有8字节无法释放并且无法调用析构
	//用delete 释放
	delete [] pp;	//因为不知道应该析构多少个A(用new会保存申请了多少个A,这时delete就会知道)
	// ,所以出现死循环析构

	return 0;
}

new A[20] 申请需要的字节
在这里插入图片描述
在这里插入图片描述

malloc(sizeof(A) * 20) 申请需要的字节

在这里插入图片描述

(2)申请一个自定义类型

//申请一个自定义类型
// 用 malloc 申请 申请空间的大小为 4字节  
A* pp = (A*)malloc(sizeof(A));	//少了构造

//用delete释放  
delete pp;	//可以但是不建议

用 new  申请  申请的空间大小为 4 字节
A* p = new A;	

free(p);	//不会报错,少了析构,当自定义类型里有需要释放的资源是就会出现内存泄漏

(3)总结:
尽量不要用这种不配对的情况,不然可能会出现各种各样的问题,最好配对使用,这样才能保证程序不会在这方面出错。

  • 29
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: malloc/free 和 new/delete 都是C/C++中的内存管理函数。malloc/free 是C语言的内存管理函数,它们用于申请释放内存空间,它们没有构造函数和析构函数,只能用于普通的内存管理。而new/deleteC++中的内存管理函数,它们用于申请释放内存空间,它们有构造函数和析构函数,可以用于类的内存管理。 ### 回答2: malloc/free和new/delete是两种动态内存分配和释放的方式。 首先,malloc/free是C语言的标准库函数,而new/deleteC++中的操作符。 其次,malloc和free是通过调用标准库函数来分配和释放内存,而newdelete是通过调用运算符来完成。 此外,malloc和free只能用于分配和释放内存,它们并不能调用对象的构造函数和析构函数。而newdelete不仅可以分配和释放内存,还可以调用对象的构造函数和析构函数。 另外,malloc分配的内存空间的大小是以字节为单位的,而new分配的内存空间的大小是以对象为单位的。因此,使用malloc分配内存时,需要手动计算所需的空间大小并进行类型转换;而使用new分配内存时,会自动计算所需的空间大小,并进行类型检查。 最后,malloc/free返回的是void指针,需要进行类型转换后才能使用;而new返回的是对象的指针,不需要进行类型转换。 综上所述,malloc/free和new/delete的区别在于语法和功能上的不同。对于C语言来说,只能使用malloc和free来进行动态内存分配和释放;而对于C++来说,则推荐使用newdelete来进行动态内存分配和释放,并能够调用对象的构造函数和析构函数。 ### 回答3: malloc/free和new/delete是两种在C和C++中用于内存分配和释放的方法。 首先,malloc和free是C语言中的函数,而newdeleteC++中的操作符。 malloc和free是库函数,用于动态分配和释放内存。它们需要手动指定要分配的内存大小,并且返回的是一个void指针。由于返回的是void指针,因此必须进行类型转换,以便使用分配的内存。此外,malloc分配的内存大小可以是0,而free不能接受空指针作为参数。 而newdeleteC++中的操作符。new操作符用于在堆上分配内存,并且会自动调用相应类型的构造函数进行对象的初始化。在使用new操作符时,不需要手动指定内存大小,而是需要指定要分配的类型。delete操作符用于释放new操作符分配的内存,并且会自动调用相应类型的析构函数进行对象的清理。同时,delete操作符还可以正确处理数组类型的对象的释放。 综上所述,malloc/free和new/delete的区别包括以下几个方面: - 使用场景:malloc/free适用于C语言,而new/delete适用于C++语言。 - 操作符与函数:malloc/free是函数,需要手动指定内存大小,返回的是void指针;而new/delete是操作符,不需要指定内存大小,可以自动调用构造函数和析构函数。 - 初始化和清理:new操作符可以自动调用构造函数进行对象初始化,而delete操作符可以自动调用析构函数进行对象清理。 - 处理数组:delete操作符可以正确处理数组类型的对象的释放。 因此,在使用C++编程时,推荐使用new/delete来替代malloc/free,可以更方便地进行内存管理和对象的初始化和清理。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值