内存分配与释放
1) new与delete的使用
A. C++中通过new关键字进行动态内存申请
B. C++中的动态内存申请是基于类型进行的
C. delete关键字用于内存释放
D. int *p = new int;delete p;//变量申请
E. char *p = new char[100];delete [] p;//数组申请
2) new关键字与malloc关键字的区别
A. new关键字是C++的运算符,malloc是由C库提供的函数
B. new以具体类型为单位进行内存分配,malloc只能以字节为单位进行内存分配
C. new返回指向此类型的指针,不用强制指针类型转换,而malloc返回指向void * 类型的指针,需要进行强制类型转换
D. new在申请单个类型变量时可进行初始化,malloc不具备内存初始化的特性
3) 内存分配示例
A. C示例:
char *p = newchar(5);
char *p1 = newchar[100];memset(p1, 0, sizeof(char) * 100);
int (*p2)[2] =new int[2][2];
int (*p3)[6][6]= new int[2][6][6];
B. C++示例:
char ptr[100] =“hello”;
charptr1[2][100] = {“hello”, “world”};
char * ptr2[3]= {“hello1”,”hello2”,”hello3”};
func(ptr);
//voidfunc(char *ptr){printf(“func = %s\n”, ptr);}
func1(ptr1);
//void func1(char*ptr[100]){int i;for(i = 0; i < 2; i++)printf(“func1 = %s\n”, ptr[i]);}
func2(ptr2);
//voidfunc2(char **ptr){int i;for(i = 0; i < 3; i++)printf(“func1 = %s\n”,ptr[i]);}
//传一维数组名用数组元素指针来接
//传二维数组名用一维数组指针来接
//传三维数组名用二维数组指针来接
//指针数组名用指针的指针来接
4) 野指针与野指针的避免
野指针:
例如:
int *p;
scanf(“%d”, p); //给p指向的内存空间赋值
printf(“%d\n”, *p);
这段代码会报段错误,原因是p为局部变量,系统给p一个随机值,p为野指针,随机指向一块内存空间,导致内存泄漏。
指向不确定地址的指针变量。(即没有初始化)使用野指针易因内存泄露出现段错误。而造成内存泄露的原因有两个:
1).访问了没有权限的内存(平时我们正确使用指针的时候,系统已经将相应的内存分配给用户,但是如果指向没有分配的内存,系统会判定我们没有权限)
2).访问了已经释放了的内存。
因为野指针主要是因为我们平时编程习惯造成的,因此我们只能避免野指针的出现,而不能杜绝。(请注意用词)我们在编程时,做到以下几点可以有效地避免野指针的出现。
第一,当一个指针没有指向时,将指针初始为NULL,降低段错误发生的可能性,也便于检查段错误。(NULL代表内存为0的地址,并且NULL是不允许做任何操作的)
第二,使用malloc分配内存。(在堆空间里分配内存)
#define MAX_SIZE 1024
char *ptr = (char *) maollc (sizeof (char) * MAX_SIZE);
请认真研究这样的表达式的优点,这个表达式在代码的维护性,扩展性都大大提高了。这方面是我们平时写代码时所应该提高的。
使用malloc也是有讲究的,我们应该依照下面的流程:
(头文件#include<stdlib.h>)
1) 分配内存。(分配成功,返回内存的首地址;分配不成功,返回NULL)。
2) 检查是否分配成功(若失败,则 exit(1) 退出程序)。
3) 清空内存中的数据(malloc分配的空间里可能存在垃圾值,因此我们需要清空,可以用到memset或bzero 函数(头文件#include <string.h>))。
4) 使用内存。
5) 释放内存(free,这时ptr又变成野指针)。
6) 写成NULL(ptr = NULL),释放地址。