new-delete误区
我们知道,new/delete是申请堆空间内存与释放堆空间内存所用的关键字。
比如,我们想要在堆空间申请4个字节,用于存放一个int型变量
int *ptr = new int;
*ptr = 10;
这时,我们为这块占用四个字节的堆空间内存,放入了一个整型变量10。到这里没有问题,我们可以理解。
当我不想用这块内存的时候,我们需要使用delete回收掉
delete ptr;
很多人会认为,执行完delete ptr,指针ptr指向的堆空间被回收,然后这段堆空间内存被清空。
这里要着重说明一下回收堆空间内存的含义:那四个字节,我在执行完delete ptr后不使用了,但是可以被别人继续使用。
new申请内存,表示的是,操作系统分配一段内存,供我现在使用。当我再次new申请内存时,可以保证操作系统分配的不是上次已经分配到的内存,操作系统只保证这么多东西。
int *ptr = new int;
*ptr = 10;
int *ptr1 = new int; //保证不是ptr指向的堆空间内存
int *ptr2 = new char; //保证不是ptr和ptr1指向的堆空间内存
但是,当我delete掉之后,就无法保证了
int *ptr = new int;
*ptr = 10;
delete ptr;
int *ptr1 = new int; //不能保证不是ptr指向的堆空间内存,意思是此时分配的内存可能时之前ptr指向的内存
int *ptr2 = new char; //保证不是ptr1指向的堆空间内存,但不能保证不是ptr指向的堆空间内存
举个例子:
int *ptr = new int();
*ptr = 10;
delete ptr;
*ptr = 20;
没有delete ptr之前的内存情况:ptr是一个指针,存储着地址为0x0000017c4e05a150的堆空间内容,该堆空间的内容为0x0000000a,也就是10。
当我delete ptr之后,ptr所在的存储空间地址没变,但是ptr所指向的存储空间变了,变成了0x0000000000008123,且该块存储空间里面的内容没有了,也就是说ptr指向了一块未知内容的内存。
我们通过ptr以前指向的地址,看看被回收之后,该块内存里是什么?
可以看到全是dd dd dd dd,也就是说delete掉的那块堆空间,回收之后里面的内容就变成了dd dd dd dd。
new/delete误区总结:
- new申请内存,操作系统会分配在堆空间上分配一段内存。当我后面继续new申请内存时,可以保证新申请的内存,不是之前申请的内存
- delete回收堆空间内存,回收的意思是:这块内存我不用了,但是别人可以继续用
- delete操作,不会清空申请的堆空间(全部赋值dd dd dd dd),不会将指向这块内存的指针清零(ptr会指向了别的存储空间)。改变指向或者清空指针,只能给指针赋值,比如ptr=nullptr;
- 申请完堆空间,立马初始化的作用就在于覆盖这段堆空间之前可能存有的值(dd dd dd dd)