c - 分配的变量不可访问

标签 c memory-management

我正在编写一个脚本,这意味着在将 PostScript 文件转换为 pdf 文件之前将文本放在末尾。

int main(int argc, char *argv[]){
    // Déclaration des variables

    // Fichier source
    char *source;
    // Extension du fichier source
    char *extension;
    // Fichier de parametre
    char *fichierPar;
    // Numero et Type de folio du fichier
    char *numeroFolio;
    char *typeFolio;

    // Tableaux contenant les types de folio a scanner
    // suivant le numero de folio
    char **t_F000 = NULL;

    // Allocation des espaces mémoires nécessaires aux variables
    source = malloc(sizeof(char) * strlen(argv[1]));
    fichierPar = malloc(sizeof(char) * strlen(argv[2]));
    extension = malloc(sizeof(char) * 5);
    typeFolio = malloc(sizeof(char) * 4);
    numeroFolio = malloc(sizeof(char) * 4);

    // Grille de parametre
    // AJUSTER LA TAILLE DU TABLEAU AVEC LE FICHIER DE LOGO.PAR
    // DANS verification.h
    Grille t_Grilles[NBGRILLE];

    // On copie les arguments dans des varibles plus explicites
    strcpy(source, argv[1]);
    strcpy(fichierPar, argv[2]);

    // GetFolio appel getNumeroFolio qui appel getTypeFolio
    // Ces deux fonctions extraient du nom de fichier
    // le numero et le type de folio
    getFolio(source, numeroFolio, typeFolio);

    printf("strlen : %d\n", strlen(numeroFolio));
    fflush(stdout);
    printf("numero : %s\n", numeroFolio);
    fflush(stdout);

    // Lecture du fichier LOGO.PAR et enregistrement en mémoire
    chargerGrilles(fichierPar, t_Grilles);

    printf("bonjour\n");
    fflush(stdout);

    printf("strlen : %d\n", strlen(numeroFolio));
    fflush(stdout);

    printf("numero : %s", numeroFolio);
    fflush(stdout);

工作并不止于此,但我正在阻止这一点。
这是 ChargerGrille() 的代码;

void chargerGrilles(char *fichierPar, Grille t_Grilles[]){
    FILE *fdParam;
    char *ligne;
    Grille *grille;
    int inc = 0;

    grille = malloc(sizeof(Grille));

    ligne = malloc(sizeof(char) * 100);

    fdParam = fopen(fichierPar, "rb");
    verifierOuverture(fdParam, fichierPar);

    while(fgets(ligne, 100, fdParam)){
        if(verifierCommentaire(ligne) != 0){

            remplirGrille(grille, ligne);
            t_Grilles[inc] = *grille;

            inc++;
        }
    }
    fclose(fdParam);
}

void remplirGrille(Grille *grille, char *ligne){
    char *split;

    // Separation sur les espaces
    split = malloc(strlen(ligne));
    split = strtok(ligne, " ");
    strcpy(grille->nom, split);
    split = NULL;
    free(split);

    split = malloc(strlen(ligne));
    split = strtok(NULL, " ");
    grille->posX = strtof(split, NULL);
    split = NULL;
    free(split);

    split = malloc(strlen(ligne));
    split = strtok(NULL, " ");
    grille->posY = strtof(split, NULL);
    split = NULL;
    free(split);

    split = malloc(strlen(ligne));
    split = strtok(NULL, " ");
    grille->longX = strtof(split, NULL);
    split = NULL;
    free(split);

    split = malloc(strlen(ligne));
    split = strtok(NULL, " ");
    grille->hautY = strtof(split, NULL);
    split = NULL;
    free(split);

    split = malloc(strlen(ligne));
    split = strtok(NULL, " ");
    strcpy(grille->logo, split);
    split = NULL;
    free(split);

    printf("Hello World !\n");
    fflush(stdout);
}

这是我的结构声明:

struct Grille{
    // Nom du tableau
    char nom[4];

    // Position du point inferieur gauche
    long int posX;
    long int posY;

    // Longueur X
    long int longX;
    // HAUTEUR Y
    long int hautY;

    // Nom du fichier logo.jpg
    char logo[50];
};

typedef struct Grille Grille;

在 gdb 中,我可以在调用 chargerGrilles(fichierPar, t_Grilles) 之前打印“source”、“extension”、“fichierPar”、“numeroFolio”和“typeFolio”;之后它将无法访问。

108             chargerGrilles(fichierPar, t_Grilles);
(gdb) print source
$7 = 0x604010 "F290390001_SCH001-2.POS"
(gdb) print numeroFolio
$8 = 0x604090 "001"
(gdb) n
Hello World !
Hello World !
Hello World !
Hello World !
110             printf("bonjour\n");
(gdb) n
bonjour
111             fflush(stdout);
(gdb) print source
$9 = 0x303732 <error: Cannot access memory at address 0x303732>
(gdb) print numeroFolio
$10 = 0x53 <error: Cannot access memory at address 0x53>

如果我在 numeroFolio 上放一个 watch ,gdb 会在这一行停止:

t_Grilles[inc] = *grille;

分配的内存发生了什么?为什么变量的地址会改变?


PS:不要管很多 printf/fflush,这是我试图找出 segFault 位置的方式

最佳答案

您面临的问题可能是由于越界访问 t_Grilles 数组到 chargerGrilles 函数引起的。使用 index > NBGRILLE 访问它会调用 Undefined Behavior并且,在您的情况下,会破坏其他主要的局部范围变量。

您必须确保循环 while(fgets(ligne, 100, fdParam)) 处理 t_Grilles 大小。

所以解决方案可以是:

 while((fgets(ligne, 100, fdParam)) && (inc <NBGRILLE)) {
    if(verifierCommentaire(ligne) != 0){

        remplirGrille(grille, ligne);
        t_Grilles[inc] = *grille;

        inc++;
    }
}

在我看来,最好的方法是向函数添加一个参数来传递数组大小。

void chargerGrilles(char *fichierPar, Grille t_Grilles[], size_t size)
{
    FILE *fdParam;
    char *ligne;
    Grille *grille;
    size_t inc = 0;

    grille = malloc(sizeof(Grille));

    ligne = malloc(sizeof(char) * 100);

    if ((grille != NULL) && (ligne != NULL))
    {
       fdParam = fopen(fichierPar, "rb");
       verifierOuverture(fdParam, fichierPar);

       while((fgets(ligne, 100, fdParam)) && (inc < size)){
          if(verifierCommentaire(ligne) != 0){

            remplirGrille(grille, ligne);
            memcpy(&t_Grilles[inc], grille, sizeof(Grille));

            inc++;
          }
       }
    }

    free(ligne);
    free(grille);
    fclose(fdParam);
}

并调用它

chargerGrilles(fichierPar, t_Grilles, sizeof(t_Grilles)/sizeof(t_Grilles[0]));

如您所见,我还添加了对 malloc 变量的检查。您总是必须检查malloc 返回值。否则你可以调用 Undefined Behavior .

最后,您必须释放 malloc内存,否则您将看到内存泄漏。

关于c - 分配的变量不可访问,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38219585/

相关文章:

C:程序即使在释放结构后仍打印垃圾文本

c++ - `n`中的 `const n = 1u;`有什么类型?

c - 如何获取用户输入的字符,并将其放置在某个内存位置?

python - 在 Cython 中创建 C 结构

matlab - 位排序算法中存储管理的改进

c++ - 内存使用率上升。释放的内存未被重用

http - 实际中空闲 HTTP 连接的理想超时长度

c - 尝试使用随机编号时 QuickSort.C 段错误。发电机

c - 使用指针 : using a global, 按值传递哪个更快?

android - 声音 fragment 的堆大小溢出问题