scanf, fscanf很相似,都是从流中读取输入,然后赋值给变量
int scanf(const char *format, ...)
int fscanf(FILE *stream, const char *format,...)
%c 单一字符
%[] 一个字符集
* 输入项读入后跳过,不赋予任何变量
scanf, fscanf 自动跳过tabs和空格,其他字符赋予变量,返回成功赋值的变量数(整数),发生错误时返回EOF
%[^ ] 表示读取一整行,^ 表示 “非”,前面提到 %[] 表示一个字符集,所以 %[^ ] 表示“遇到一个换行符之前的所有字符的集合”,随后就结束读入。
char str[80];
fscanf(fp,"%[^ ]",str);
但是这么做以后,那个换行符还没有读取。
前面提到了,* 表示输入项读入后跳过,所以 %*c 表示跳过一个字符,所以可以写
fscanf(fp, "%[^ ]%*c", str);
读取换行符之前的所有内容,赋值给str,然后跳过换行符,指向下一行内容。
如果要跳过整行,可以写
fscanf(fp, "%*[^ ]%*c");
但有时也会出错,当缓存中只有一个 时,%*[^ ] 发现 之前没有任何内容,会直接返回,后面的 %*c 不起作用。所以缓存中只有 时,上面那行代码不管运行多少遍,都是什么都不做。
为了避免上面的情况,可以修改为
fscanf(fp, "%*c%*[^ ]");
那么如果缓存中只有一个 ,%*c 会跳过它,%*[^ ] 会跳过下一行换行符之前的内容;如果缓存中不是 ,则 %*c 跳过第一个字符,%*[^ ] 跳过后面的内容(换行符之前),即跳过一行,缓存中留一个 。
缓存中留一个 不影响 %d, %lf 等语句,所以这么写更安全。