7.2 常见的内存错误及其对策
问题1: 内存分配未成功,却使用了它
解决办法:在使用内存之前检查指针是否为NULL。如果指针p是函数的参数,那么在函数的入口处用assert(p!=NULL)进行检查。如果是用malloc或new来申请内存,应该用if(p==NULL)或if(p!=NULL)进行防错处理。
问题2: 内存分配虽然成功,但是尚未初始化就引用它。
问题3: 内存分配成功并且已经初始化,但操作越过了内存的边界
问题4: 忘记了释放内存,造成内存泄漏
问题5: 释放了内存却继续使用它
【规则7-2-1】用malloc 或new 申请内存之后,应该立即检查指针值是否为NULL。防止使用指针值为NULL 的内存。
【规则7-2-2】不要忘记为数组和动态内存赋初值。防止将未被初始化的内存作为右值使用。
【规则7-2-3】避免数组或指针的下标越界,特别要当心发生“多1”或者“少1”操作。
【规则7-2-4】动态内存的申请与释放必须配对,防止内存泄漏。
【规则7-2-5】用free 或delete 释放了内存之后,立即将指针设置为NULL,防止产生“野指针”。
7.3 指针与数组的对比
7.3.1 修改内容
示例7-3-1 中,字符数组a 的容量是6 个字符,其内容为hello/0。a 的内容可以改变,如a[0]= ‘X’。指针p 指向常量字符串“world”(位于静态存储区,内容为world/0),常量字符串的内容是不可以被修改的。从语法上看,编译器并不觉得语句p[0]= ‘X’有什么不妥,但是该语句企图修改常量字符串的内容而导致运行错误。
char a[] = "hello";
a[0] = 'X';
cout << a << endl;
char *p = "world"; // 注意p 指向常量字符串
p[0] = 'X'; // 编译器不能发现该错误,运行时出错
cout << p << endl;
cin.get();
7.3.2 内容复制与比较
不能对数组名进行直接复制与比较。示例7-3-2 中,若想把数组a 的内容复制给数组b,不能用语句 b = a ,否则将产生编译错误。应该用标准库函数strcpy 进行复制。同理,比较b 和a 的内容是否相同,不能用if(b==a) 来判断,应该用标准库函数strcmp进行比较。
语句p = a 并不能把a 的内容复制指针p,而是把a 的地址赋给了p。要想复制a的内容,可以先用库函数malloc 为p 申请一块容量为strlen(a)+1 个字符的内存,再用strcpy 进行字符串复制。同理,语句if(p==a) 比较的不是内容而是地址,应该用库函数strcmp 来比较。
// 数组…
char a[] = "hello";
char b[10];
strcpy(b, a); // 不能用b = a;
if(strcmp(b, a) == 0) // 不能用if (b == a)
…
// 指针…
int len = strlen(a);
char *p = (char *)malloc(sizeof(char)*(len+1));
strcpy(p,a); // 不要用p = a;
if(strcmp(p, a) == 0) // 不要用if (p == a)
…
示例-3-2 数组和指针的内容复制与比较
7.3.3 计算内存容量
用运算符sizeof 可以计算出数组的容量(字节数)。示例7-3-3(a)中,sizeof(a)的值是12(注意别忘了’/0’)。指针p 指向a,但是sizeof(p)的值却是4。这是因为sizeof(p)得到的是一个指针变量的字节数,相当于sizeof(char*),而不是p 所指的内存容量。C++/C 语言没有办法知道指针所指的内存容量,除非在申请内存时记住它。
注意当数组作为函数的参数进行传递时,该数组自动退化为同类型的指针。
示例7-3-3(b)中,不论数组a 的容量是多少,sizeof(a)始终等于sizeof(char *)。
char a[] = "hello world";
char *p = a;
cout<< sizeof(a) << endl; // 12 字节
cout<< sizeof(p) << endl; // 4 字节
示例-3-3(a)计算数组和指针的内存容量
void Func(char a[100])
{
cout<< sizeof(a) << endl; // 4 字节而不是字节
}
示例-3-3(b)数组退化为指针