7.6 错误处理(stderr 和 exit)
cat
程序的错误处理功能并不完善
如果因为某种原因而造成其中的一个文件无法访问,相应的诊断信息要在该连接的输出的末尾才能打印出来
当输出到屏幕时,这种处理方法尚可以接受,但如果输出到一个文件或通过管道输出到另一个程序时,就无法接受了
为了更好地处理这种情况,另一个输出流以与 stdin
和 stdout
相同的方式分派给程序,即 stderr
即使对标准输出进行了重定向,写到 stderr
中的输出通常也会显示在屏幕上
下面我们改写 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);
}
该程序通过两种方式发出出错信息
首先,将 fprintf
函数产生的诊断信息输出到 stderr
上,因此诊断信息将会显示在屏幕上,而不是仅仅输出到管道或输出文件中
诊断信息中包含 argv[0]
中的程序名,因此,当该程序和其它程序一起运行时,可以识别错误的来源
其次,程序使用了标准库函数 exit
,当该函数被调用时,它将终止调用程序的执行
任何调用该程序的进程都可以获取 exit
的参数值,因此,可通过另一个将该程序作为子进程的程序来测试该程序的执行是否成功
按照惯例,返回值 0
表示一切正常,而非 0
返回值通常表示出现了异常情况
exit
为每个已打开的输出文件调用 fclose
函数,以将缓冲区中的所有输出写到相应的文件中
在主程序 main
中,语句 return expr
等价于 exit(expr)
但是,使用函数 exit
有一个优点,它可以从其它函数中调用,并且可以用类似于第 5 章中描述的模式查找程序查找这些调用
如果流 fp
中出现错误,则函数 ferror
返回一个非 0
值
int ferror(FILE *fp)
尽管输出错误很少出现,但还是存在的(例如,当磁盘满时),因此,成熟的产品程序应该检查这种类型的错误
函数 feof(FILE *)
与 ferror
类似,如果指定的文件到达文件结尾,它将返回一个非 0
值
int feof(FILE *fp)
在上面的小程序中,我们的目的是为了说明问题,因此并不太关心程序的退出状态
但对于任何重要的程序来说,都应该让程序返回有意义且有用的值