我正在编写一个脚本,这意味着在将 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/