c - 使用 C 将 CSV 文件解析为结构

标签 c csv struct strsep

我正在尝试解析 CSV 文件并将值放入结构中,但是当我退出循环时,我只返回文件的值。我无法使用 strtok,因为 csv 文件中的某些值是空的并且它们会被跳过。我的解决方案是 strsep,当我在第一个 while 循环内时,我可以打印所有歌曲,但当我离开时,它只会返回文件的最后一个值。

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

#define BUFFSIZE 512

typedef struct song{
    char *artist;
    char *title;
    char *albumName;
    float duration;
    int yearRealeased;
    double hotttness;
} song;

typedef song *songPtr;

 int main(){
     FILE *songStream;
     int count = 0;
     int size = 100;
     char *Token;

     char *buffer;
     song newSong;
     songPtr currentSong;
     songPtr *allSongsArray = malloc(size * sizeof(songPtr));
     buffer = malloc(BUFFSIZE+1);

     songStream = fopen("SongCSV.csv", "r");

     if(songStream == NULL){
         err_sys("Unable to open file");
     }else{
         printf("Opened File");
             while(fgets(buffer, BUFFSIZE, songStream) != NULL){
                 char *line = buffer;
                 int index = 0;

                 while ((Token = strsep(&line,","))) {
                     if(index == 17){
                         newSong.title = malloc(sizeof(Token));
                         newSong.title = Token;
                     }
                     index++;
                 }
                 currentSong = malloc(1*sizeof(song));
                 *currentSong = newSong;
                 allSongsArray[count] = malloc(sizeof(currentSong) + 1);
                 allSongsArray[count] = &newSong;
                 free(currentSong);
                 printf("\n%s\n", allSongsArray[count]->title);
                 count++;

                 if(count == size){
                     size = size + 100;
                     allSongsArray = realloc(allSongsArray ,size * sizeof(songPtr));
                 }
             }
             fclose(songStream);
     }

     fprintf(stdout,"Name in struct pointer array:  %s\n",allSongsArray[2]->title);


     return 0;
 }

有人可以告诉我为什么会发生这种情况以及如何解决吗?

最佳答案

您应该将 newSong.title = Token; 更改为 strcpy(newSong.title,Token); 因为 Token 指向缓冲区中的一个地址,该地址将在读取下一行时保存新数据

此外,它还可以避免内存泄漏

newSong.title = malloc(sizeof(Token));

只会分配 sizeof(char *) 字节,因此您分配的字节数少于您需要的字节数,如果 token 超过 4 或 8 个字节,则可能会导致分段,具体取决于系统上的 sizeof(char *)

             *currentSong = newSong; 
             allSongsArray[count] = malloc(sizeof(currentSong) + 1); 
             allSongsArray[count] = &newSong;
             free(currentSong);

更改为

memcpy(currentSong,&newSong,sizeof(newSong));
allSongsArray[count] = currentSong;

关于c - 使用 C 将 CSV 文件解析为结构,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46210630/

相关文章:

C,转换通过 void 指针传递的结构数组

c - 逐行读取txt文件并处理每个单词

php - 使用 PHP 代码和 HTML 表单将 excel (.csv) 导入 MySQL

mysql - 使用 LOAD DATA INFILE 从 CSV 文件导入 MySQL 中的数据

xml - 将 XML 转换为 CSV 不起作用

c - C 中程序的双重释放或损坏 (!prev) 错误

c - 未初始化的值是由堆栈分配创建的 - valgrind

c++ - 他们最擅长什么——C 和 C++

c - 分配固定字符数组与指针以转换为字符串

c++ - 从文件中读取信息