一:问题描述:
出现的问题如下:
二:问题产生的原因说明
该问题发生于操作堆内存的时候。产生该问题的原因是:你实际使用的内存大小超出了你实际申请的内存大小,在释放内存的时候就会发生该问题。
举个例子:假如你申请了3个字节的堆内存空间 char *ptr = (char *)malloc(sizeof(char)*3);
但是你在使用的时候使用了4个字节,char *t1 = "abc";// 注意末尾有一个'\0'字符
strcpy(ptr, buf);// 拷贝函数默认会在末尾添加一个'\0'字符,在此处你仅仅申请了3个字节的空间,但你实际使用的空间为4个字节
在往ptr指向的3个字节的堆内存空间,写入超过一个字节的内容时, vs2010中的编译器并不会报错, 但是为后面的内存释放埋下个巨大的隐患。
此后当你在释放ptr指向的内存时,由于你使用了超过申请大小的内存空间,导致内存不能正确释放,就会发生上面的问题。
扫描二维码关注公众号,回复:
15346947 查看本文章
三:举例
有问题代码如下:
#define _CRT_SECURE_NO_WARNINGS
#include <iostream>
using namespace std;
int main()
{
const char* t1 = "abc";
cout << sizeof(t1) << endl; // 4
// 申请内
int memSize = 3;
char* ptr = (char*)malloc(sizeof(char) * memSize); // 仅仅申请了3个字节的堆内存空间
// 初始化内存
memset(ptr, 0, memSize);
// 拷贝内容
strcpy(ptr, t1); // strcpy拷贝时末尾默认会添加一个'\0'字符, 所以实际使用的内存空间为4个字节,超出了申请的内存空间大小,为后来的内存释放埋下了巨大的隐患
cout << ptr << endl;
if (ptr != NULL)
{
free(ptr);
ptr = NULL;
}
system("pause");
return 0;
}
上面的代码运行后就会出现下面的问题
正确代码:
#define _CRT_SECURE_NO_WARNINGS
#include <iostream>
using namespace std;
int main()
{
const char* t1 = "abc";
cout << sizeof(t1) << endl; // 4
// 申请内
//int memSize = 3;// 申请内存的大小必须与实际使用的内存大小一致
int memSize = 4;// 实际使用的内存大小为4个字节,那么申请内存必须大于等于实际使用的内存。
char* ptr = (char*)malloc(sizeof(char) * memSize);
// 初始化内存
memset(ptr, 0, memSize);
// 拷贝内容
strcpy(ptr, t1);
cout << ptr << endl;
if (ptr != NULL)
{
free(ptr);
ptr = NULL;
}
system("pause");
return 0;
}
四:总结
当实际使用的内存大小超过了申请内存空间大小时,在以后释放内存的时候就会引发HEAP CORRUPTION DETECTED:after Normal block错误。
解决方案:当申请堆内存空间的时候,申请的空间大小一定要大于等于实际上使用的内存空间大小。