printf、fprintf 和 freopen() 进行重定向

一、printf 和 fprintf

除了人工的分析之外,最简单最直接的调试方法要算printf了。不过,我们这里推荐使用的并不是初学 C 语言时使用的函数

int printf(const char *format, ...)

而是稍微复杂一点的 fprintf() 函数,因为它更方便我们之后重定向错误输出信息到指定的设备。fprintf() 函数的原型如下:

int fprintf(FILE *stream, const char *format, ...)

可以看到,它与 printf() 函数相比多出来了第一个参数 FILE *stream,其意义是将打印的内容输出到文件流指针 stream 指向的流。所谓流,通常是指程序输入或输出的一个连续的字节序列,设备(例如鼠标、键 盘、磁盘、屏幕、调制解调器和打印机)的输入和输出都是用流来处理的,在 C 语言中,所有的流均以文件的形式出现 —— 不一定是物理磁盘文件,还可以是对应于某个输入/输出源的逻辑文件。C 语言提供了 5 种标准的流,你的程序在任何时候都可以使用它们,并且不必打开或关闭它们。以下列出了这5种标准的流。

------------------------------------------------
    名称          描  述             例  子
------------------------------------------------
    stdin        标准输入             键盘
    stdout       标准输出             屏幕
    stderr       标准错误              屏幕
    stdprn       标准打印机          LPT1端口
    stdaux       标准串行设备       COM1端口
------------------------------------------------
其中,stdprn 和 stdaux 并不总是预先定义好的,因为 LPT1 和 COM1 端口在某些操作系统中是没有意义的,而 stdin、stdout 和 stderr 总是预先定义好的。此外,stdin 并不一定来自键盘,stdout 也并不一定显示在屏幕上,它们都可以重定向到磁盘文件或其它设备上。我们在头文件 stdio.h 中可以找到 stdin、stdout 和 stderr 的定义如下:

/* Standard streams.  */

extern struct _IO_FILE *stdin;      /* Standard input stream.  */

extern struct _IO_FILE *stdout;     /* Standard output stream.  */

extern struct _IO_FILE *stderr;     /* Standard error output stream.  */

在使用 fprintf() 函数时,通常我们可以将第一个参数设为 stdout 或者 stderr,打印出错调试信息的时候则推荐使用 stderr 而不是 stdout,这是一种惯例,同时也由于内核在处理 stdout 和 stderr 时的优先级不一样,后者的优先级要高一些,因此有时候如果程序异常退出 时,stderr 能得到输出,而 stdout 就不行。

printf(...) 实际上相当于 fprintf(stdout, ...),这也是为什么我们不推荐使用它的原因。在输出调试信息的时候,我们推荐使用 fprintf(stderr, …),或者使用某个指定的文件流 fprintf(some_stream, …)。

那么具体如何在必要的时候重定向 fprintf() 中的调试信息呢?来看看下面的一些方法:

当调试信息的量比较大,需要一些时间或者其他辅助工具来搜索过滤时,仅仅利用显示屏幕来输出调试信息是不够的,这时我们经常将这些信息输出到所谓的日志文件(log)中,之后再仔细的分析log文件来发现问题。

二、利用 Shell 的 I/O 重定向

简单的写 log 方法可以通过 shell 的 I/O 重定向机制来实现,比如下面的代码:

#include <stdio.h>

int main()
{
    fprintf(stdout, "This is a standard output info;\n");
    fprintf(stderr, "This is a error output info;\n");
    return 0;
}

在默认条件下,编译运行的结果是打印信息都输出在屏幕上:

This is a standard output info;
This is a error output info;

这是因为默认情况下,shell 所打开的 stdout 和 stderr 设备都是显示屏幕。

三、用 freopen() 进行重定向

有时候我们要求在程序中能够控制标准流的重定向,这时可以利用标准 C 库函数 freopen()。freopen() 的函数原型如下:

FILE *freopen(const char *filename, const char *mode, FILE *stream)

下面的代码用来测试用函数 freopen() 重定向 stderr:

#include <stdio.h>

int main()
{
    if (freopen("err.log", "w", stderr) == NULL) {
        fprintf(stderr, "error redirecting stderror!");
        return -1;
    }
    fprintf(stdout, "This is a standard output info;\n");
    fprintf(stderr, "This is a error output info;\n");
    fclose(stderr);
    return 0;
}

maina 开始时 freopen() 函数将 stderr 重定向到了”err.log”文件,这样得到的结果如下:

$ gcc -o fprint fprint.c 

$ ./fprint

This is a standard output info;

$ cat ./err.log

This is a error output info;

可见打印到stderr的信息被重定向到了 err.log 文件中,而 stdout 的打印信息则还是输出到了屏幕上。

四、拓展

1、fscanf

按“格式字符串”所指定的格式,从“文件类型指针”所指向的文件的当前位置读取数据,然后按“输入项地址表列”的顺序,将读取来的数据存入指定的内存单元中。

fscanf 函数的返回值是读取的数据个数;若遇见文件结束符或读取不成功,则 fscanf 函数返回 EOF(-1)。

例如:

fscanf(fp,“%d,%f”,&i,&t);

表示从 fp 所指向的文件中,按“%d,%f”规定的格式读取两个值,将这两个值分别存储在地址 &i 和 &t 对应的内存单元中。若读取成功,fscanf 函数的返回值是 2 。

即:

fscanf 从 fp 中按照格式读取数据。

fprintf 将多个字符按照某个格式写入 fp 中。

转载:printf和fprintf及用freopen()进行重定向_mydriverc2的博客-CSDN博客_fprintf 重定向

(SAW:Game Over!) 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值