realloc 调用后内存泄漏

如何解决realloc 调用后内存泄漏

我有一个 **void,其中每个位置都指向 *void,该指针指向我的 struct 记录,其中包含 3 个字段(*char、int、float)。我想从结构中的 csv 加载数据,但是当需要重新分配内存时,因为大小等于数组的容量,我得到了 realloc(): invalid next size。我没有得到我的 printf("realloc failed") 所以我认为 tmp 不为空,但无论如何我丢失了我的内存指针。

struct record{
    int id;
    char* field1;
    int field2;
    float field3; 
};

long array_size = 0; 
long array_capacity = INITIAL_CAPACITY;
void** array;

void** array_create(){

    void **array = (void**)malloc(INITIAL_CAPACITY*sizeof(void*));
    if(array == NULL){
        fprintf(stderr,"array_create: unable to allocate memory for the array");
        exit(EXIT_FAILURE);
    }

    return array;
}

void add_on_array(void** array,void* elem){

    if(array == NULL){
        fprintf(stderr,"add_on_array: array parameter cannot be NULL");
        exit(EXIT_FAILURE);
    }
    if(elem == NULL){
        fprintf(stderr,"add_on_array: elem parameter cannot be NULL");
        exit(EXIT_FAILURE);
    }
    
    if(array_size >= array_capacity){

        void** tmp = realloc(array,2*(sizeof(void*)*array_capacity));
        if(tmp == NULL){
            printf("Realloc fails\n");
            exit(EXIT_FAILURE);
        }
        array = tmp;

        array_capacity = 2*array_capacity;
    }
    array[array_size] = elem;
    array_size++;

}

    while(fgets(buffer,buf_size,fp) != NULL){
        read_line_p = strdup(buffer);
        if(read_line_p == NULL){
            fprintf(stderr,"main: unable to allocate memory for the read line");
            exit(EXIT_FAILURE);
        }

        // strcpy(read_line_p,buffer);
        char *id_field_in_read_line_p = strtok(read_line_p,",");
        char *field1_in_read_line_p = strtok(NULL,"); 
        char *field2_in_read_line_p = strtok(NULL,");  
        char *field3_in_read_line_p = strtok(NULL,");

        char *field1 = malloc((strlen(field1_in_read_line_p)+1)*sizeof(char));
        
        if(field1 == NULL){
            fprintf(stderr,"main: unable to allocate memory for the string field of the read record");
            exit(EXIT_FAILURE);
        }
        int id = atoi(id_field_in_read_line_p); 
        strcpy(field1,field1_in_read_line_p); 
        int field2 = atoi(field2_in_read_line_p);
        float field3 = atof(field3_in_read_line_p);
        struct record *record_p = malloc(sizeof(struct record));
        if(field1 == NULL){
            fprintf(stderr,"main: unable to allocate memory for the read record");
            exit(EXIT_FAILURE);
        } 
        record_p->id = id;
        record_p->field1 = field1;
        record_p->field2 = field2;
        record_p->field3 = field3;
        add_on_array(array,(void*)record_p);
        
        free(read_line_p);
    }
    fclose(fp);
    stampa_array(array,array->size);
    printf("\nData loaded\n");
    
}

解决方法

好吧,你在这里做了一些“非常糟糕”的事情。

首先,这里有一个名为 array 的全局变量:

void** array;

一般来说,全局变量是你应该避免的。 (在极少数情况下,您确实需要一个全局变量,我建议您给它一个不用于其他任何事物的“丑陋”名称 - 例如:globalVariableArray

但更糟糕的是 - 您还有一个函数,这里有一个名为 array 的参数:

void add_on_array(void** array,void* elem){
    ...
}

这是什么意思?同时拥有全局 array 和参数 array?哪些会在函数中访问?

答案是函数参数 array 充当函数局部变量,它将隐藏全局变量。

所以当你这样做时:

array = tmp;

更改本地 array 变量 - 而不是全局变量。

当函数返回时,局部变量不再存在,即对其所做的任何更改都将丢失。

换句话说 - realloc 分配的内存丢失了,你有泄漏。

解决此问题的第一步是:

  1. 将全局变量移动到 main

  2. 调用需要改变array的函数时,需要传递array的地址,即&array。功能原型必须相应更改。

对于其他全局变量也是如此...

但是为什么不把所有的数组内容放到一个结构体中呢?

喜欢:

struct ArrayContainer
{
    long array_size; 
    long array_capacity;
    void** array;
}

main 中:

struct ArrayContainer container = {0,INITIAL_CAPACITY,NULL};
container.array = ...;  // Allocate INITIAL_CAPACITY

我认为这将大大简化您的代码,因为您只需要将指向此结构的指针传递给函数。然后您可以通过该指针更改所有三个成员。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


依赖报错 idea导入项目后依赖报错,解决方案:https://blog.csdn.net/weixin_42420249/article/details/81191861 依赖版本报错:更换其他版本 无法下载依赖可参考:https://blog.csdn.net/weixin_42628809/a
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下 2021-12-03 13:33:33.927 ERROR 7228 [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPL
错误1:gradle项目控制台输出为乱码 # 解决方案:https://blog.csdn.net/weixin_43501566/article/details/112482302 # 在gradle-wrapper.properties 添加以下内容 org.gradle.jvmargs=-Df
错误还原:在查询的过程中,传入的workType为0时,该条件不起作用 <select id="xxx"> SELECT di.id, di.name, di.work_type, di.updated... <where> <if test=&qu
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct redisServer’没有名为‘server_cpulist’的成员 redisSetCpuAffinity(server.server_cpulist); ^ server.c: 在函数‘hasActiveC
解决方案1 1、改项目中.idea/workspace.xml配置文件,增加dynamic.classpath参数 2、搜索PropertiesComponent,添加如下 <property name="dynamic.classpath" value="tru
删除根组件app.vue中的默认代码后报错:Module Error (from ./node_modules/eslint-loader/index.js): 解决方案:关闭ESlint代码检测,在项目根目录创建vue.config.js,在文件中添加 module.exports = { lin
查看spark默认的python版本 [root@master day27]# pyspark /home/software/spark-2.3.4-bin-hadoop2.7/conf/spark-env.sh: line 2: /usr/local/hadoop/bin/hadoop: No s
使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams['font.sans-serif'] = ['SimHei'] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -> systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping("/hires") public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate<String
使用vite构建项目报错 C:\Users\ychen\work>npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-