Jar*_*mis 2 c file scanf fgets
我需要从文件中读取文本,并根据读取的信息为结构赋值.
以下是文本文件的格式:
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
Run Code Online (Sandbox Code Playgroud)
结构我必须把它读成:
typedef struct
{
char first[8], initial, last[10],
street[17], city[12], state[3];
int age, tenure, zip;
float salary;
char sex;
}Employee;
Run Code Online (Sandbox Code Playgroud)
我到目前为止使用的代码是:
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);
}
Run Code Online (Sandbox Code Playgroud)
但是当我打印每一行时,前几行很好,然后事情开始变得混乱:
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
Run Code Online (Sandbox Code Playgroud)
我也尝试过:
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);
}
Run Code Online (Sandbox Code Playgroud)
但是这给出了相同的结果.
代码不能在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;
}
Run Code Online (Sandbox Code Playgroud)
这用于%c读取单个字符; 它用于%16c读取多字街道地址并%11c读取(可能是多字)城市.它使用%7s,%9s和%2s防止其他字符串溢出.fscanf()呼叫后的分配确保计数的%c字符串为空终止; 本身,%16c不添加空终止符.
该inputLine()函数现在-1在出现问题时返回错误指示(已选择),并0指示成功.这与UNIX系统调用一个共同的模式,但是从底层的行为不同scanf()的功能-家庭,作为一个著名的评论通过chux.
此代码使用fscanf()问题中的内容读取标准输入.它确保emp阵列也没有溢出.
#include <stdio.h>
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);
}
Run Code Online (Sandbox Code Playgroud)
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
Run Code Online (Sandbox Code Playgroud)
此代码用于fgets()读取行和sscanf()转换数据.使用此版本的代码可以更容易地报告错误.
#include <stdio.h>
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);
}
Run Code Online (Sandbox Code Playgroud)
对于问题中的样本数据,此输出与另一个的输出相同.