C-从文本文件

时间:2016-06-13 20:15:11

标签: c string printf

我在以正确格式打印信息时遇到问题。我的程序应该读取有关航班预订的信息。信息应作为行程打印出来,如下:

    ******************* Flight Reservation Request  **********************
petem@xyz.net Pete Moss (M 1986/01/01)
123 Boggy Lane
New Orleans, LA 70112
   Flight       Seats      
   H100.15005      2      
   H222.15005      2
   H200.15010      2
   H333.15010      2       
******************* Flight Reservation Request  **********************
pcorn@abc.net Pop Corn (M 1957/02/02)
456 Kernel
San Antonio, TX 78210
   Flight       Seats      
   H222.15005      1
   HXXX.XXXXX      1
   H333.15010      1    

以下是大多数重要的代码。我只在processReservations()工作。

#define _CRT_SECURE_NO_WARNINGS 1

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include "cs1713p1.h"
FILE *pFileCust;               // stream Input for Customer Reservation data

void processCommandSwitches(int argc, char *argv[], char **ppszCustomerFileName);
void processReservations();


int main(int argc, char *argv[])
{
    char *pszCustomerFileName = NULL;

    // Process the command switches
    processCommandSwitches(argc, argv,  &pszCustomerFileName);

    // open the Customer Reservation stream data file
    if (pszCustomerFileName == NULL)
        exitError(ERR_MISSING_SWITCH, "-c");

    pFileCust = fopen(pszCustomerFileName, "r");
    if (pFileCust == NULL)
        exitError(ERR_CUSTOMER_RES_FILENAME, pszCustomerFileName);

    // process the Reservations
    processReservations();

    fclose(pFileCust);
    printf("\n");    // included so that you can put a breakpoint on this line
    return 0;
}

/****** Where my errors are coming from *****/

void processReservations()
{

Customer customer;                        // student structure for holding student data
FlightRequest flight;
char szInputBuffer[100];  // input buffer for fgets
int iScanfCnt;                          // scanf returns the number of successful inputs


  // read data input lines of text until EOF.  fgets returns NULL at EOF
    while (fgets(szInputBuffer, 100, pFileCust) != NULL)
    {
        // if the line is just a line feed, skip it.
        if (szInputBuffer[0] == '\n')
            continue;

        iScanfCnt = sscanf(szInputBuffer, "%1c %10s %50s %30[^\n] %50[^,] %2s %2s %5s %10s %d\n"
            , &customer.cGender
            , customer.szBirthDt
            , customer.szEmailAddr
            , customer.szFullName
            , customer.szStreetAddress
            , customer.szCity
            , customer.szStateCd
            , customer.szZipCd
            , flight.szFlightId
            , &flight.iRequestSeats);

        // Check for bad input.  scanf returns the number of valid conversions
        if (iScanfCnt < 10)
        {
            printf("invalid input when reading data, only %d valid values. \n"
                , iScanfCnt);
            printf("\tdata is %s\n", szInputBuffer);
            //return ERR_MISSING_SWITCH;
        }

        printf("******************* Flight Reservation Request  **********************\n");
        printf("%s %s", customer.szEmailAddr, customer.szFullName);
        printf("(%c %s)", customer.cGender, customer.szBirthDt);
        printf("%s, %s %s\n", customer.szStreetAddress, customer.szCity, customer.szZipCd);
        printf("   Flights  Seats\n");
        printf("   %s       %d\n", flight.szFlightId, flight.iRequestSeats);
    }
    fclose(pFileCust);
}

以下是第一位客户的部分输出结果:

invalid input when reading data, only 4 valid values. 
    data is M 1986/01/01 petem@xyz.net Pete Moss

******************* Flight Reservation Request  **********************
petem@xyz.net Pete Moss
(M 1986/01/01), a a
   Flights  Seats
invalid input when reading data, only 4 valid values. 
    data is 123 Boggy Lane,New Orleans,LA,70112

******************* Flight Reservation Request  **********************
Boggy Lane,New Orleans,LA,70112
(1 23), a a
   Flights  Seats
显然很长很麻烦,但我对C很陌生,不得不弄清楚出了什么问题。一个问题是“航班预订请求”行的打印方式太多次了。每次从输入文件中读取数据变量时,似乎while循环从头开始重复,我不知道如何解决这个问题。我在另一个网站上发现了类似的问题,一位用户建议

  

使用临时变量/结构来测试我所在的输入文件的哪一行,所以每次我只测试我在文件中的位置时,我都不会覆盖我的最终客户结构。另一个想法是根据我可以从输入文件中做出的假设来构建我的代码。例如,如果我已成功读取新客户的第一行,那么我知道下一行将遵循其他格式,因此我不必检查每个后续行是否是新客户的第一行。

我不明白这些提议的解决方案中的任何一个意味着什么,或者如何修复这些代码,但是我已经在很长一段时间里一直在努力,没有任何进展。 我的所有变量声明都包含一个.h文件,但每个变量名前面的小写字母都表示该变量的数据类型(sz = string,c = char,i = int)。有什么问题?

1 个答案:

答案 0 :(得分:1)

程序报告错误

  

读取数据时输入无效,只有4个有效值

这是因为输入缓冲区

char szInputBuffer[100];

太小了。

中字段宽度限制的总和
iScanfCnt = sscanf(szInputBuffer, "%1c %10s %50s %30[^\n] %50[^,] %2s %2s %5s %10s %d\n", ...);

更大。所以很可能

while (fgets(szInputBuffer, 100, pFileCust) != NULL)

会将每行输入分成几个较小的字符串,这些字符串是您未检测到的(缺少尾随newline)。但是sscanf确实失败了。

不要吝啬。使用慷慨的缓冲区,如

char szInputBuffer[4096];