我正在努力学习Zed A. Shaw的艰难方式ex25
#include <stdlib.h>
#include <stdio.h>
#include <stdarg.h>
#include "dbg.h"
#define MAX_DATA 100
int read_string(char **out_string, int max_buffer)
{
*out_string = calloc(1, max_buffer + 1);
check_mem(*out_string);
char *result = fgets(*out_string, max_buffer, stdin);
check(result != NULL, "Input error.");
return 0;
error:
if (*out_string) free(*out_string);
*out_string = NULL;
return -1;
}
int read_int(int *out_int)
{
char *input = NULL;
int rc = read_string(&input, MAX_DATA);
check(rc == 0, "Failed to read number.");
*out_int = atoi(input);
free(input);
return 0;
error:
if (input) free(input);
return -1;
}
int read_scan(const char *fmt, ...)
{
int i = 0;
int rc = 0;
int *out_int = NULL;
char *out_char = NULL;
char **out_string = NULL;
int max_buffer = 0;
va_list argp;
va_start(argp, fmt);
for (i = 0; fmt[i] != '\0'; i++) {
if (fmt[i] == '%') {
i++;
switch (fmt[i]) {
case '\0':
sentinel("Invalid format, you ended with %%.");
break;
case 'd':
out_int = va_arg(argp, int *);
rc = read_int(out_int);
check(rc == 0, "Failed to read int.");
break;
case 'c':
out_char = va_arg(argp, char *);
*out_char = fgetc(stdin);
break;
case 's':
max_buffer = va_arg(argp, int);
out_string = va_arg(argp, char **);
rc = read_string(out_string, max_buffer);
check(rc == 0, "Failed to read string.");
break;
default:
sentinel("Invalid format.");
}
} else {
fgetc(stdin);
}
check(!feof(stdin) && !ferror(stdin), "Input error.");
}
va_end(argp);
return 0;
error:
va_end(argp);
return -1;
}
int main(int argc, char *argv[])
{
char *first_name = NULL;
char initial = ' ';
char *last_name = NULL;
int age = 0;
printf("What's your first name? ");
int rc = read_scan("%s", MAX_DATA, &first_name);
check(rc == 0, "Failed first name.");
printf("What's your initial? ");
rc = read_scan("%c\n", &initial);
check(rc == 0, "Failed initial.");
printf("What's your last name? ");
rc = read_scan("%s", MAX_DATA, &last_name);
check(rc == 0, "Failed last name.");
printf("How old are you? ");
rc = read_scan("%d", &age);
printf("---- RESULTS ----\n");
printf("First Name: %s", first_name);
printf("Initial: '%c'\n", initial);
printf("Last Name: %s", last_name);
printf("Age: %d\n", age);
free(first_name);
free(last_name);
return 0;
error:
return -1;
}
我对line 109
中read_scan中的第一个参数感到困惑原始代码运行正常。输出:
What's your first name? zed
What's your initial? A
What's your last name? shaw
How old are you? 18
---- RESULTS ----
First Name: zed
Initial: 'A'
Last Name: shaw
Age: 18
但是,如果我删除line 109 rc = read_scan("%c", &initial);
中的'\ n',它将跳过下一个问题,我无法理解。
我对'\ n'影响的看法是for loop不会进入line 83 fgetc(stdin);
输出将是:
What's your first name? zed
What's your initial? A
What's your last name? How old are you? shaw
---- RESULTS ----
First Name: zed
Initial: 'A'
Last Name:
Age: 0
感谢您的帮助!
答案 0 :(得分:1)
"%c\n"
到read_scan()
中的字符表示读取并捕获一个字符,然后读取并丢弃另一个字符。您可以使用X
或@
代替\n
,它的工作方式相同。从格式中删除换行符时,字符后面的换行符将保留在输入中。然后,使用read_scan()
调用"%s"
会调用fgets()
,它会读取下一个换行符,但下一个换行符已经在输入流中,因此会立即返回。
请注意,如果你输入一个单词而不是一个单词,或者如果你没有中间的首字母(我!),那么就会出现不同的错误。
答案 1 :(得分:0)
第一个参数是read_scan()
需要处理的格式字符串的类型和大小
rc = read_scan("%c\n", &initial);
在这种情况下,它告诉read_scan()
读取(a)任何字符和(b)换行符。如果查看函数中的流,它会检查“%”,如果找到它会递增计数器i
,然后检查格式字符串中的下一个字符。对于第109行,该字符为c
,因此调用fgetc()
函数。这将从文件描述符stdin
中读取单个字符并将其放入out_char
。然后循环继续,因为我们没有读取字符串,所以我们转到第82行的else
情况 - 它读取另一个字符(换行符为'\ n')。与第83行fgetc
调用的重要区别在于我们丢弃了结果。