jockchou

C语言常用的输入输出函数


函数原型如下:

#include <stdio.h>

int puts(const char *s);
char *gets(char *s);

int fputs(const char *s, FILE *stream);
char *fgets(char *s, int size, FILE *stream);

int putc(int c, FILE *stream);
int getc(FILE *stream);

int fputc(int c, FILE *stream);
int fgetc(FILE *stream);

int putchar(int c);
int getchar(void);

函数功能描述

puts() 输出一个字符串到标准输出流stdout,并在字符串结尾处自动添加换行符。

gets() 从标准输入设备读取一行字符保存到s指向的缓冲地址中,直到终端输入一个换行符或者读到EOF,并将换行符替换成一个空字符('\0')。它不会检查目标缓冲区是否能够容纳输入,所以很不安全。不要使用这个函数,应该使用fgets代替它。

fputs() fputs()函数是puts的面向文件的版本,第二个参数用来说明输出的文件,可以使用stdout作为来调用,stdout在stdio.h中定义。与puts()不同,fputs()并不为输出自动添加换行符,也不会输出结尾的空字符。

fgets() gets()函数有一个bug,它不检查预留存储区是否足够容纳输入的内容,fgets()改进了这个问题,它第二个参数让你指定最大读入字符数。同时fgets()会把读到的换行符存到字符串里。第三个参数用来指定读取哪一个文件,从键盘读数据时,可以使用stdin,这个标识符在stdio.h中定义。

fputc() 输出一个字符C到指定文件中,会将C转换成无符号字符类型。

fgetc() 从指定文件中读取下一个字符,返回字符的整数表示,或者是EOF,或者是错误。

putc() putc()同等于fputc()函数,如果被实现成宏的形式,它的流不能是有副作用的表达式。

getc() putc()同等于fputc()函数,如果被实现成宏的形式,它的流不能是有副作用的表达式。

putchar() 等同于putc(c, stdout)。

getchar() 等同于getc(stdin)。

从以上说明我们可以看出,我们不需要记住那么多的函数,只需要使用fgetc(),fputc()他fgets()和fputs()这2组基本可以满足开发需求。其他的要不就是这2组的宏封装,要不就是有Bug,可被替代。同时记住fgets()函数第二个参数用来指定读取的最大长度,输出超过这个长度或者遇到换行或者文件尾,就会返回。fgets()会保留读取的换行,fputs()不会自动添加换行。这样正好匹成一对。把文件指定为stdin或者stdout即可达到输出到标准io设备的效果。

int fputs(const char *s, FILE *stream);
char *fgets(char *s, int size, FILE *stream);

int fputc(int c, FILE *stream);
int fgetc(FILE *stream);

注意: 有副作用的表达式,指的是表达式执行后,会改变表达式中某些变量的值,例如(++, --)操作。这样的表达式不应该在宏里出现,否则在宏展开时候会出现意想之外的结果。例如计算平方的宏:

#include<stdio.h>

#define MACRO_SQRT(x) (x)*(x)       
int func_sqrt(int x)       
{
    return x * x;       
}

int main(void)
{
    int x, y, n = 5, m = 5;
    x = MACRO_SQRT(++n);
    y = func_sqrt(++m);

    printf("MACRO_SQRT = %d\n", x); //49
    printf("func_sqrt  = %d\n", y); //36

    return 0;
}

宏与函数的区别在于,函数的参数是如果是表达式,会先计算出表达式的结果,再传递给函数使用。而宏是先展开,再计算表达式的值。宏展开后是这个样子的:

(++n)* (++n)

2次++后,n的值为7,结果便是7*7=49。为了避免这种问题,在给宏传递参数时,不要使用有副作用的表达式,即表达式中变量值会变化的表达式。

代码举例

#include<stdio.h>

int main(void)
{    
    /*假设想输入1234567890这10个字符,
    * 多出的2个空间用来保存输入的换行符和自动添加的空字符(\0)
    */
    char input_str[12];

    fputs("input your name, please!\n", stdout);

    fgets(input_str, sizeof(input_str) / sizeof(char), stdin);

    fputs(input_str, stdout);

    return 0;
}

测试输出:

$ ./fputs 
input your name, please!
1234567890
1234567890

$ ./fputs 
input your name, please!
1234567890aaaa
1234567890a$

当多输出aaaa这几个字符时,输出最后的一个换行符被第一个a占据了,也就是说input_str数组中保存了'1234567890a\0'这12个字符。也就是说char *fgets(char *s, int size, FILE *stream)函数第二个参数如果指定为12,系统最多读取11个字符,并自动添加一个'\0'。

上面的程序有一个缺点,即fgets()函数会自动保存输入的换行,但是如果空间不够,换行就不会有,字符串里换行符(\n)的不确定性会给显示带来不方便,正如上例中那样。一个好的解决办法是,检测fgets()输入的字符串,去掉最后的换行符,在输出需要时,再考虑添加换行符。

#include<stdio.h>
#include<string.h>

int main(void)
{    

    char *pos;

    char input_str[11];

    fputs("input your name, please!\n", stdout);

    fgets(input_str, sizeof(input_str) / sizeof(char), stdin);

    if (pos = strchr(input_str, '\n')) {
        *pos = '\0';
    }

    printf("%s\n", input_str);

    return 0;
}

上例展示了使用strchr()函数来定位输入的字符串最后是否有换行符,如果有,则把它去掉。