C从文件中读取以空格分隔的值

我需要从文件中读取文本,并根据读取的信息为结构赋值。

这是文本文件的格式:

First Middle Last Address city state zip age sex tenure salary \-------------------------------------------------------------- ADA A AGUSTA 33 BABBAGE ROAD LOVELACE GB 19569 28 F 2 350.50 ISSAC A ASIMOV 99 FICTION WAY AMHERST MA 63948 58 M 6 423.88 HUMPHRY R BOGART 71 SAM STREET HOLLYWOOD CA 48482 56 M 5 366.00 

结构我必须把它读成:

 typedef struct { char first[8], initial, last[10], street[17], city[12], state[3]; int age, tenure, zip; float salary; char sex; }Employee; 

我到目前为止使用的代码是:

 void inputLine(Employee* e) { fscanf(pay, "%s %s %s %s %s %s %s %s %d %d %s %d %f", e->first, &e->initial, e->last, e->street, e->street, e->city, e->city, e->state, &e->zip, &e->age, &e->sex, &e->tenure, &e->salary); } 

但是当我打印每一行时,前几行很好,然后事情开始变得混乱:

 TED L KOPPEL ABC WASHINGTON DC 37376 48 M 9 909.44 DAVID T LETTERMAN WNBC NEW YORK 0 0 NY 1 47 5 STEVIE R 0 0 NICKS 3 MUSIC CHICAGO 23459 38 0 0 F 8 460.88 P 76 SILLY 0 89 STREET L GB 44 2 320.50 0 12341 

我也尝试过:

 void inputLine(Employee* e) { char line[53]; fgets(line, 52, pay); printf("%s\n", line); fflush(stdout); sscanf(line, "%s %s %s %s %s %s %s %s %d %d %s %d %f", e->first, &e->initial, e->last, e->street, e->street, e->city, e->city, e->state, &e->zip, &e->age, &e->sex, &e->tenure, &e->salary); } 

但是这给出了相同的结果。

代码不能在C ++中。 我正在使用Linux GNU GCC 4.8.1编译器,但我也在Mac的编译器上测试了它,但它没有用。

全球变量,如pay – 明确定义为FILE *pay; – 是一个坏主意,在示例代码中完全没有必要。 始终测试fscanf()及其亲属的返回值,以确保获得所需的所有数据。

但是,您的麻烦是%s在第一个空格处停止,因此您在阅读地址字段时遇到了巨大的问题。 您的输入也不受约束。 您还尝试多次使用e->street获取街道地址的多个单词; 这是行不通的,因为第三个单词会覆盖第一个单词。

你需要这样的东西:

 int inputLine(FILE *fp, Employee* e) { if (fscanf(fp, "%7s %c %9s %16c %11c %2s %d %d %c %d %f", e->first, &e->initial, e->last, e->street, e->city, e->state, &e->zip, &e->age, &e->sex, &e->tenure, &e->salary) != 11) return -1; e->street[16] = '\0'; e->city[11] = '\0'; return 0; } 

这使用%c来读取单个字符; 它使用%16c读取多字街道地址,使用%11c读取(可能是多字)城市。 它使用%7s%9s%2s来防止其他字符串溢出。 fscanf()调用后的赋值确保计数的%c字符串为空终止; %16c本身不添加空终止符。

inputLine()函数现在在出现问题时返回错误指示(选择-1 ), 0表示成功。 这是Unix系统调用的常见模式,但与基础scanf() – 函数系列的行为不同,如chux的注释中所述。

工作代码1

此代码使用fscanf()作为问题读取标准输入。 它确保了emp数组也没有溢出。

 #include  typedef struct { char first[8], initial, last[10], street[17], city[12], state[3]; int age, tenure, zip; float salary; char sex; } Employee; void dump_employee(FILE *fp, const char *tag, const Employee *e); int inputLine(FILE *fp, Employee *e); enum { MAXEMP = 10 }; int main(void) { char line[4096]; Employee emp[MAXEMP]; if (fgets(line, sizeof(line), stdin) == 0 || fgets(line, sizeof(line), stdin) == 0) return 1; for (int i = 0; i < MAXEMP && inputLine(stdin, &emp[i]) != 0; i++) dump_employee(stdout, "Employee", &emp[i]); return 0; } int inputLine(FILE *fp, Employee *e) { if (fscanf(fp, "%7s %c %9s %16c %11c %2s %d %d %c %d %f", e->first, &e->initial, e->last, e->street, e->city, e->state, &e->zip, &e->age, &e->sex, &e->tenure, &e->salary) != 11) return -1; e->street[16] = '\0'; e->city[11] = '\0'; return 0; } void dump_employee(FILE *fp, const char *tag, const Employee *e) { fprintf(fp, "%s: %-7s %c %-9s %-16s %-11s %-2s %.5d %3d %c %d %6.2f\n", tag, e->first, e->initial, e->last, e->street, e->city, e->state, e->zip, e->age, e->sex, e->tenure, e->salary); } 

样本输出

 Employee: ADA A AGUSTA 33 BABBAGE ROAD LOVELACE GB 19569 28 F 2 350.50 Employee: ISSAC A ASIMOV 99 FICTION WAY AMHERST MA 63948 58 M 6 423.88 Employee: HUMPHRY R BOGART 71 SAM STREET HOLLYWOOD CA 48482 56 M 5 366.00 

工作代码2

此代码使用fgets()读取行,使用sscanf()来转换数据。 使用此版本的代码可以更容易地报告错误。

 #include  typedef struct { char first[8], initial, last[10], street[17], city[12], state[3]; int age, tenure, zip; float salary; char sex; } Employee; void dump_employee(FILE *fp, const char *tag, const Employee *e); int scan_employee(Employee *e, const char *line); enum { MAXEMP = 10 }; int main(void) { char line[4096]; Employee emp[MAXEMP]; if (fgets(line, sizeof(line), stdin) == 0 || fgets(line, sizeof(line), stdin) == 0) return 1; for (int i = 0; i < MAXEMP && fgets(line, sizeof(line), stdin) != 0; i++) { if (scan_employee(&emp[i], line) == 0) dump_employee(stdout, "Employee", &emp[i]); } return 0; } int scan_employee(Employee *e, const char *line) { if (sscanf(line, "%7s %c %9s %16c %11c %2s %d %d %c %d %f", e->first, &e->initial, e->last, e->street, e->city, e->state, &e->zip, &e->age, &e->sex, &e->tenure, &e->salary) != 11) return -1; e->street[16] = '\0'; e->city[11] = '\0'; return 0; } void dump_employee(FILE *fp, const char *tag, const Employee *e) { fprintf(fp, "%s: %-7s %c %-9s %-16s %-11s %-2s %.5d %3d %c %d %6.2f\n", tag, e->first, e->initial, e->last, e->street, e->city, e->state, e->zip, e->age, e->sex, e->tenure, e->salary); } 

对于问题中的样本数据,此输出与另一个的输出相同。