改写 cat 程序,将其出错信息写到标准错误文件上:
#include <stdio.h>
/* cat: concatenate files, version 2 */
main(int argc, char *argv[])
{
FILE *fp;
void filecopy(FILE *, FILE *);
char *prog = argv[0]; /* program name for errors */
if (argc == 1 ) /* no args; copy standard input */
filecopy(stdin, stdout);
else
while (--argc > 0)
if ((fp = fopen(*++argv, "r")) == NULL) {
fprintf(stderr, "%s: can't open %s\n", prog, *argv);
exit(1);
} else {
filecopy(fp, stdout);
fclose(fp);
}
if (ferror(stdout)) {
fprintf(stderr, "%s: error writing stdout\n", prog);
exit(2);
}
exit(0);
}
1.stderr
为了更好处理因为某种原因而造成文件无法访问,另一个输出流以与 stdin 和 stdout 相同的方式分派给程序,即 stderr。即使对标准输出进行了重定向,写到 stderr 中的输出通常也会显示在屏幕上。
上面的程序将 fprintf 函数产生的诊断信息输出到 stderr 上,因此诊断信息将会显示在屏幕上,而不是仅仅输出到管道或输出文件中。诊断信息中包含 argv[0]中的程序名,因此,当该程序和其它程序一起运行时,可以识别错误的来源。
2.exit
标准库函数 exit,当该函数被调用时,它将终止调用程序的执行:
- 任何调用该程序的进程都可以获取 exit 的参数值,因此,可通过另一个将该程序作为子进程的程序来测试该程序的执行是否成功。一般来说,返回值 0 表示一切正常,而非 0 返回值通常表示出现了异常情况。
- exit 为每个已打开的输出文件调用 fclose 函数,以将缓冲区中的所有输出写到相应的文件中。
- 在主程序 main 中,语句 return expr 等价于 exit(expr)。 但是,使用函数 exit 有一个优点,它可以从其它函数中调用,并且可以用类似于模式查找程序查找这些调用。
3.ferror
int ferror(FILE *fp)
如果流 fp 中出现错误,则函数 ferror 返回一个非 0 值。 尽管输出错误很少出现,但还是存在的(例如,当磁盘满时),因此,成熟的产品程序应该检查这种类型的错误。
4.feof
int feof(FILE *fp)
函数 feof(FILE *)与 ferror 类似。如果指定的文件到达文件结尾,它将返回一个非0 值。