[转]HEAP CORRUPTION DETECTED:after Normal block错误方法解决

一:问题描述:

出现的问题如下:

二:问题产生的原因说明

  该问题发生于操作堆内存的时候。产生该问题的原因是:你实际使用的内存大小超出了你实际申请的内存大小,在释放内存的时候就会发生该问题。

  举个例子:假如你申请了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错误。

解决方案:当申请堆内存空间的时候,申请的空间大小一定要大于等于实际上使用的内存空间大小。

猜你喜欢

转载自blog.csdn.net/laolitou_1024/article/details/125740381