【问题标题】:Reading a multiline text file using sscanf and fgets使用 sscanf 和 fgets 读取多行文本文件
【发布时间】:2016-09-24 23:46:21
【问题描述】:

我需要读取一个文本文件,然后将结果打印出来。我使用结构来保存变量,并且不需要创建数组。问题是代码似乎要么没有读取第二行,要么我打印不正确,结果完全混乱。这是文本文件:

UTSA01 8 0 150.00 80.00
Armadillos,utsa@xyz.com,(210)555-1111,Jean E Us 
COM001 7 1 150.00 75.00
Comm Eagles,maeking@xyz.com,(210)555-2222,Mae King
SOUTH1 5 3 120.00 75.00
Slam Dunk,slamdunk@gmail.com,(210)555-3333,Jerry Tall
ALHGHT 4 4 175.00 100.00
Cake Eaters,sliverspoon@xyz.com,(210)555-6666,E Z Street
UNKN01 1 7 150.00 50.00 
Org New Blk,bobwire@xyz.com,(210)555-1234,Bob Wire
NEWB01 0 8 120.00 75.00
River Rats,riverrat@xyz.com,(210)555-4444,Rock D Boat
UNKN02 3 5 150.00 75.00
Hackers,cyber@gmail.com,(210)555-5555,Tom E Gunn

我一直在 while 循环中使用 sscanf 和 fgets。这是我的代码:

char szInputBuffer[100];
/* Create a struct array that will hold the data that is being read in */
Team teams;

/* The following segment contains the header for the file */
printf("%-7s%-13sWins Loss Fee Amt  Paid Amt\n", "Id", "Team Name");
printf("                %-21s%-14sEmail\n", "Contact Name", "Phone");

/* The following structure will begin to read file data into the respectful 
   variables until the end-of-file marker has been reached.
   The data will be used later to print out. */

while(!feof(pFileTeam)) {
    fgets(szInputBuffer, 100, pFileTeam);
    sscanf(szInputBuffer, "%7s %d %d %lf %lf",
                           teams.szTeamId,
                           &teams.iWins,
                           &teams.iLosses,
                           &teams.dFeeAmount,
                           &teams.dPaidAmount);

    fgets(szInputBuffer, 256, pFileTeam);                      
    sscanf(szInputBuffer, "%[^,]%[^,]%[^,]%[^\n]",
           teams.szTeamName,
           teams.szEmailAddr,
           teams.szPhone,
           teams.szContactname);


    printf("%-7s%-13s%-5d%-5d%-9.2lf%-8.2lf\n",
        teams.szTeamId,
        teams.szTeamName,
        teams.iWins,
        teams.iLosses,
        teams.dFeeAmount,
        teams.dPaidAmount);
    printf("                %-21s%-14s%-5s\n",
        teams.szContactname,
        teams.szPhone,
        teams.szEmailAddr);
}//END while

我遇到的问题是我可以读取和打印文件的第一行,第二行中的第一个字符串很长。但是,一旦我点击逗号,它似乎就会被绊倒。我不确定它是我在 sscanf 中使用的说明符还是正在打印。我真的很困惑我该怎么做。

这是我编译 (gcc) 并执行时的结果:

Id     Team Name    Wins Loss  Fee Amt   Paid Amt
                    Contact Name         Phone         Email
    UTSA01 Armadillos      8    0    150.00     80.00

    COM001 Comm Eagles     7    1    150.00     75.00

    SOUTH1 Slam Dunk       5    3    120.00     75.00

    ALHGHT Cake Eaters     4    4    175.00    100.00

    UNKN01 Org New Blk     1    7    150.00     50.00

    NEWB01 River Rats      0    8    120.00     75.00

    UNKN02 Hackers         3    5    150.00     75.00

它实际上在我的终端的第二行显示了一些奇怪的字符,但它们似乎没有出现在这里。

编辑:我已经添加了逗号,但输出仍然无法正常工作。只有最后打印的行会显示 szContactname 信息。所有其他人都将该位置留空,但会显示电话和电子邮件:

Id     Team Name    Wins Loss  Fee Amt   Paid Amt
                    Contact Name         Phone         Email
    UTSA01 Armadillos      8    0    150.00     80.00
                    (210)555-1111 utsa@xyz.com
    COM001 Comm Eagles     7    1    150.00     75.00
                    (210)555-2222 maeking@xyz.com
    SOUTH1 Slam Dunk       5    3    120.00     75.00
                    (210)555-3333 slamdunk@gmail.com
    ALHGHT Cake Eaters     4    4    175.00    100.00
                    (210)555-6666 sliverspoon@xyz.com
    UNKN01 Org New Blk     1    7    150.00     50.00
                    (210)555-1234 bobwire@xyz.com
    NEWB01 River Rats      0    8    120.00     75.00
                    (210)555-4444 riverrat@xyz.com
    UNKN02 Hackers         3    5    150.00     75.00
                    Tom E Gunn           (210)555-5555 cyber@gmail.com

【问题讨论】:

  • 您可能还有其他问题,但对于初学者:Why is “while ( !feof (file) )” always wrong?
  • 我添加了逗号,但最后一行是唯一正确的。其他行都打印电话号码和电子邮件,但没有联系人姓名。但是,最后一行以正确的格式打印了联系人姓名、电话和电子邮件。

标签: c text struct fgets scanf


【解决方案1】:

一些问题

  1. 使用fgets() 缓冲区而不检查结果。在尝试使用szInputBuffer 之前检查fgets() 的结果。进一步feof() 不需要检查。 @kaylum

    // while(!feof(pFileTeam)) {
    //    fgets(szInputBuffer, 100, pFileTeam);
    //    sscanf(szInputBuffer, ....
    while(fgets(szInputBuffer, sizeof szInputBuffer, pFileTeam) != NULL) {
        sscanf(szInputBuffer, ....
    
  2. 使用幻数。想想 256。szInputBuffer[] 没那么大。

    char szInputBuffer[100];
    ...  
    // fgets(szInputBuffer, 256, pFileTeam);   
    if (fgets(szInputBuffer, sizeof szInputBuffer, pFileTeam) == NULL) break;
    
  3. 无用的格式。 "%[^,]%[^,]..." 尝试读取直到 ',' 的文本,并且 离开 ',' 在文件流中而不读取它。 @user3121023。健壮的代码限制字符串输入并检查sscanf() 结果。

    // sscanf(szInputBuffer, "%[^,]%[^,]%[^,]%[^\n]", teams.szTeamName, ...
    // Assuming szTeamName[10]
    if (4 != sscanf(szInputBuffer, "%9[^,],%9[^,],%9[^,],%9[^\n]", teams.szTeamName, ...) {
      break;
    }
    
  4. 可能是其他问题。

【讨论】:

  • 好的,所以我使用了你的建议,它仍然像以前一样运行。我明白你在说什么,这真的很有帮助。但是,我的结果仍然让我感到困惑,因为我不明白对 sscanf 的检查如何解析为 4,并且仍然不包含一个变量的数据。更令人困惑的是,前 6 组变量为空白,但最后一组包含信息。
  • 所以我更改了控制换行符的正则表达式,只使用了一个可以接受字符和空格的正则表达式——我知道这并不理想——它奏效了。感谢您为我指明了正确的方向,这真的很有帮助!
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2019-07-25
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2021-11-09
  • 2017-08-29
  • 1970-01-01
相关资源
最近更新 更多