【问题标题】:Freeing up memory allocated for a very long string (char*) in C?在 C 中释放为非常长的字符串 (char*) 分配的内存?
【发布时间】:2015-07-07 15:05:45
【问题描述】:

请,我需要帮助来释放分配的内存以用于一个非常长的字符串。我试图将代码缩减为包含我遇到的问题的这个小片段:

#include <stdlib.h>
#include <string.h>
#include <math.h>
#include <stdio.h>
#include <limits.h>

#define N 6
#define E 1024

int M[N][N][N], R[N*N][N], C[N*N][N], F[N*N][N];

void init_rcf() {
    int i, j, k, p;

    p = 0;
    for(j=0; j<N; j++) {
        for(k=0; k<N; k++) {
            for(i=0; i<N; i++) 
                R[p][i] = M[i][j][k];
            p++;
        }
    }

    p = 0;
    for(i=0; i<N; i++) {
        for(k=0; k<N; k++) {
            for(j=0; j<N; j++)
                C[p][j] = M[i][j][k];
            p++;
        }
    }

    p = 0;
    for(i=0; i<N; i++) {
        for(j=0; j<N; j++) {
            for(k=0; k<N; k++)
                F[p][k] = M[i][j][k];
            p++;
        }
    }
}

char *bin(int n, int p) {
    int c, d, count;
    char *pointer;
    count = 0;
    pointer = (char*)malloc(p+1);

    for (c = p-1;c >= 0;c--) {
        d = n >> c;
        if (d & 1) 
            *(pointer+count) = 1 + '0';
        else 
            *(pointer+count) = 0 + '0';
        count++;
    }
    *(pointer+count) = '\0';
    return  pointer;
}

int f0(int n) {
    return ceil(log2(n+1)); 
}

int f1() {
    int sum, max = 0;
    for(int k=0;k<N;k++) {
        for(int j=0;j<N;j++) {
            for(int i=0;i<N-1;i++) {
                sum = M[i][j][k] + M[i+1][j][k]; 
                if (sum > max) 
                    max = sum; 
            }
        }   
    }
    return max;
}

int f2() {
    int sum, max = 0;
    for(int k=0;k<N;k++) {
        for(int i=0;i<N;i++) {
            for(int j=0;j<N-1;j++) {
                sum = M[i][j][k] + M[i][j+1][k]; 
                if (sum > max) 
                    max = sum; 
            }
        }   
    }
    return max;
}

int f3() {
    int sum, max = 0;
    for(int j=0;j<N;j++) {
        for(int i=0;i<N;i++) {
            for(int k=0;k<N-1;k++) {
                sum = M[i][j][k] + M[i][j][k+1]; 
                if (sum > max) 
                    max = sum; 
            }
        }   
    }
    return max;
}

int f4() {
    int m1 = f1(),  m2 = f2(), m3 = f3(); 
    if ((m1 >= m2) && (m1 >= m3)) return m1;
    if ((m1 <= m2) && (m2 >= m3)) return m2;
    if ((m1 <= m3) && (m2 <= m3)) return m3;
}

char *g_fxn() {
    char *g = (char *) malloc(1 + (N*N*N)*3); 
    int k = f0(f4());

    init_rcf();
    strcpy(g,"");            
    for(int i=0; i<N*N; i++) 
        for(int j=0; j<N; j++) 
            strcat(g,bin(R[i][j],k));
    for(int i=0; i<N*N; i++) 
        for(int j=0; j<N; j++) 
            strcat(g,bin(C[i][j],k));
    for(int i=0; i<N*N; i++) 
        for(int j=0; j<N; j++) 
            strcat(g,bin(F[i][j],k));
    return g;
}

void convert2bin(char *file_in) {
    const char *FILENAME_IN = file_in;
    const char FILENAME_OUT[] = "temp.txt";
    char c, d; 
    int k=0;

    FILE *infile;
    FILE *outfile;

    infile = fopen(FILENAME_IN, "rb");
    outfile = fopen(FILENAME_OUT, "w");
    if(infile == NULL){
        fprintf(stderr, "Error: Source file not found.\n");
        exit(EXIT_FAILURE);
    }

    while((c = fgetc(infile)) != EOF){
        k++;
        unsigned n = (sizeof(c) * CHAR_BIT) - 1; 
        for (int i=0; i<=n; i++) { 
            int m = (c >> (n-i)) & 1; 
            d = '0'+m;          
            fwrite(&d, 1, sizeof(d), outfile);
        }
    }

    if (k < E) {
        d = '1';
        fwrite(&d, 1, sizeof(d), outfile);
        for (int i=k; i<=E; i++) { 
            d = '0';            
            fwrite(&d, 1, sizeof(d), outfile);
        }
    }

    fclose(infile);
    fclose(outfile);
}

void digest() {
    const char *FILENAME_IN = "temp.txt";
    const char FILENAME_OUT[] = "digest.txt";

    int size = N*N*N; // size of message to construct N*N*N matrix
    char c, msg0[size], *msgf; 

    FILE *infile;
    FILE *outfile;

    infile = fopen(FILENAME_IN, "r");
    outfile = fopen(FILENAME_OUT, "wb");
    if(infile == NULL){
        fprintf(stderr, "Error: Source file \"temp.txt\" not found.\n");
        exit(EXIT_FAILURE);
    }

    int quit = 0;
    while (quit == 0) {
        msgf = (char *) malloc(1 + (size)*3);
        strcpy(msgf, ""); 
        strcpy(msg0, "");
        int p = 0;
        while(((c = fgetc(infile)) != EOF) && (p < size)) {
            msg0[p++] = c;
        }

        if(c == EOF) quit = 1;
        if (p > 0) {
            if (p < size) {
                msg0[p] = '1';
                for(int i=p+1; i<size; i++) 
                    msg0[i] = '0';
            }

            for (int k=0; k<N; k++)
                for (int j=0; j<N; j++)
                    for (int i=0; i<N; i++) {
                        c = msg0[i + N * (j + N * k)];
                        if (c == '0')
                            M[i][j][k] = 0;
                        else 
                            M[i][j][k] = 1;
                    }

            strcpy(msgf, g_fxn());             
            int q = 0;

            while (q<strlen(msgf)) {
                int d;
                char b = 0;
                for (int r=0; r<8; r++) {
                    if (msgf[q++] == '0') 
                        d = 0;
                    else 
                        d = 1;
                    b = ((b<<1) | d);
                }
                fwrite(&b, 1, 1, outfile);
                b = 0;
            }
        }
    }

    free(msgf);
    fclose(infile);
    fclose(outfile);
}

int main(int argc, char *argv[]){
    if (argc!=2) {
        fprintf(stderr, "Error: Provide name of one source file.\n");
        exit(EXIT_FAILURE);
    }

    char *clear_file = argv[1];

    convert2bin(clear_file);

    digest();

    printf("File successfully digested!\n");

    return(0);
}

此代码适用于最多 27 个字节的输入文件,但除此之外它每次都会崩溃。

在尝试调试后,我发现问题在于 strcpy(msgf, g_fxn()); 对函数 g_fxn 的调用,我的猜测是那里的问题来自为消息分配的大块内存,但这似乎以后不得解脱。我猜这是因为代码在对 g_fxn 的调用较少但调用较多时崩溃。

有人对我如何解决这个问题有任何想法吗?

【问题讨论】:

  • 首先总是cjeck malloc 返回。如果为 NULL,则未分配您的指针。
  • 你的缩进很混乱!
  • convert2bin() 中的c 更改为int c 并重写unsigned n = (sizeof(c) * CHAR_BIT) - 1;
  • 缩进代码时,切勿使用制表符,因为每个编辑器/字处理器对制表符宽度/制表符停止位都有不同的设置。建议每行只放置一个代码语句,以便我们人类阅读
  • 发布的代码无法编译。当询问运行时问题时,发布干净编译的代码。宏 CHAR_BIT 未定义(因为相应的头文件尚未#include'd。函数 f4() 的路径不以 'return value;' 语句结尾。这一行:'while (q

标签: c string memory-management crash malloc


【解决方案1】:

msgf 在函数 digest() 的 while 循环中被分配了很多次:

msgf = (char *) malloc(1 + (size)*3);

但是free()只在循环之后被调用一次:

free(msgf);

这是我发现的至少一个内存泄漏。

【讨论】:

  • bin() 函数似乎也分配了从未释放的内存。
  • bin()函数不需要释放内存,也可以在使用返回值(字符串)后释放。为此,您必须将返回的指针存储在缓冲区变量中,然后使用缓冲区变量并随后释放它(每次调用 bin() 时)。
  • @RhinoDevel 谢谢你的评论,你是对的,但我不能这样做。在 while 循环中释放 msgf - 就在关闭循环之前 - 即使对于 1 字节的输入文件,它也会崩溃!
  • @DavidLaPorte 至于我遇到最大问题的bin() 函数中分配的内存,我无法释放g,因为这正是函数返回的内容。我不知道如何释放bin() 函数中分配的内存...
  • @RhinoDevel 感谢您的建议。但是有一个问题:这是否意味着我将不得不使用全局变量作为缓冲区?我问这个是因为我认为否则我将无法释放它。
【解决方案2】:

有些函数是分配了内存但没有像中那样释放 -

char *g_fxn()char *bin(int n, int p)

编辑

在这些函数中返回指针时,您可以使用goto 语句。

请看下面的例子,它只是展示了如何解决你的问题 -

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

char lol(char c[]);

char lol(char c[])
{ 
  char a[6]="hello";
  char *p=malloc(sizeof(a));
  p=a;
  memcpy(c,p,strlen(a));
  goto DONE;

  DONE:
  free(p);
  return c;
}

int main()
{
   char b[5];
   lol(b);
   printf("%s",b);
  return 0;
}

【讨论】:

  • 感谢@ameyCU,这正是我遇到问题的地方:在g_fxn() 中,在while 循环中释放为msgf 分配的内存使其崩溃,而在bin() 中我不知道如何释放为我要返回的变量分配的内存?!
  • @IT_guy 请检查编辑。我尝试解决您的问题,您可以通过修改在您的程序中使用类似的逻辑。
  • 感谢这个想法,我会探索它并看看它是如何进行的,虽然我不太确定如何在我的代码中调整它。不过,我会考虑一下。
【解决方案3】:

如果一个函数分配内存并返回结果指针,而不是释放内存,那么调用函数必须确保它在不再需要时被释放。在您的 strcpy(msgf, g_fxn()) 案例中,可能如下所示:

char *tmp = g_fxn();
strcpy(msgf, tmp);
free(tmp);

通常,每次您从malloc() 获得指向内存的指针时,您必须稍后将指向该内存的指针传递给free(),一旦不再需要它。

【讨论】:

  • 感谢@sth,我也试过了,但在这种情况下它会因 1 字节输入文件而崩溃。
猜你喜欢
  • 2018-04-19
  • 2015-07-07
  • 2020-11-28
  • 2017-05-22
  • 1970-01-01
  • 1970-01-01
  • 2011-06-11
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多