• C语言的标准输入输出


    欢迎探讨,如有错误敬请指正 

    如需转载,请注明出处 http://www.cnblogs.com/nullzx/


    1. 标准输入输出

    clip_image002标准输入、输出主要由缓冲区和操作方法两部分组。缓冲区实际上可以看做内存中的字符串数组,而操作方法主要是指printf、scanf、puts、gets,getcha、putcahr等操作缓冲区的方法。在C++以及Java等面向对象的编程语言中,将缓冲区以及操作缓冲区的方法封装成一类对象,这类对象就称为流。

    缓冲区最大的特点主要体现在数据的一次性,即数据被printf、scanf从缓冲区中取出后就被使用了,或者说消耗了。可以把缓冲区比喻成管道,缓冲区中的数据比喻成水流,printf、scanf等方法比喻成开关,当打开开关,水就会慢慢流逝,而流出去的水就再也收不回来了。

    由于不同系统,不的硬件底层实现输入输出的具体方法可能不一样,C语言要求系统为每个程序提供两个指针,这两个指针分别指向两个结构体,这两个结构体分别表示了键盘和屏幕在内存中的抽象表示(缓冲区的地址值被记录在这个结构体中),并将指向这两个结构体的指针命名为stdin和 stdout.这两个指针就是所谓的标准输入和标准输出。

    还有一点应该始终铭记,标准输入和输出缓冲区中存储的是字符的ASCII码值。比如你想从键盘上输入了123给一个变量,那么在缓冲区中存储是三个字节,分别是字符‘1’的ASCII码值,字符‘2’的ASCII码值,字符‘3’的ASCII码值,然后将这个这三个ASCII值序列转换为一个数值给这个变量。同理,从屏幕输出“123”,计算机并不认为它输出的是一个数值,计算机实际上仅仅是描绘了一个‘1’的ASCII码值对应的图形,‘2’的ASCII的值对应的图形,‘3’的ASCII码值对应的图形。

    2. getchar、putchar

    putchar的作用主要是向输出缓冲区中写入一个字符。

    getchar的作用主要是向输入缓冲区中读取一个字符。如果碰到文件结尾,返回-1

    getchar源代码

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    int getchar(void){
        static char buf[BUFSIZ];
        static char* bb = buf;
        static int n = 0;
        if (n == 0) {
            n = read(0, buf, BUFSIZ);
            bb = buf;
        }
        return(--n >= 0) ? (unsigned char)*bb++ : EOF;
    }

    OEF是一个宏,表示-1。getchar的返回值是int,对于文件来说-1表示了文件的结尾。我们可以在键盘上利用Ctrl+Z来实现类似的效果

    从getchar的源代码中可以看出,如果发现字符数组buf已空(n==0),则调用read方法从键盘读取数据(该方法会导致阻塞),并让指针指向数组的首地址。如果缓冲区还有字符没有被读取(n > 0),则读取它,同时n-1,指针(bb)向后移动一位。当缓冲区已空(n==0),且read函数读取失败时(读取到了文件末尾),返回EOF。

    从scanf的源代码中可以看出getchar可以读入任何字符,包括空白符(空白符包括:空格、换行符、制表符等)。

    3. gets、puts

    puts函数主要向输出缓冲区写入一个字符串,并再字符串输出结束以后,再额外输出一个换行符 'n'

    gets用于从输入流的缓冲区中读取字符到指定的数组。读取过程中会忽略所有的前导空白符,读入的第一个字符为非空白符,直到遇到换行符才停止读入,结束的换行符('n')被gets函数读从缓冲区读取走了,存于数组中,然后被替换成'0'

    gets 源代码(只需要看for循环这部分代码,FLOCKFILE(stdin)表示对输入缓冲区加锁对;FUNLOCKFILE(stdin) 表示对输入缓冲区解锁)。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    char* gets(char *buf){
        int c;
        char *s;
        static int warned;
        static const char w[] = "warning: this program uses gets(), which is unsafe. ";
        FLOCKFILE(stdin); ORIENT(stdin, -1);
        if (!warned) {
            (void)_write(STDERR_FILENO, w, sizeof(w) - 1);
            warned = 1;
        }
        for (s = buf; (c = __sgetc(stdin)) != ' ';)
            if (c == EOF)
                if (s == buf) {
                     FUNLOCKFILE(stdin); return (NULL);
                } else
                    break;
            else
                *s++ = c; *s = 0;
            FUNLOCKFILE(stdin); return (buf);
    }

    从源代码可以看出,如果读入了'n'则停止,并替换成''

    4. printf的使用

    定义函数 int printf(const char * format,...);

    函数说明 printf()会根据参数format字符串来转换并格式化数据,然后将结果写出到标准输出设备,直到出现字符串结束('n')为止。

    参数format字符串可包含下列三种字符类型:

    (1)一般文本,伴随直接输出。

    (2)转义字符,如 、 等。

    (3)格式转换字符,格式转换为一个百分比符号(%)及其后的格式字符所组成。一般而言,每个%符号在其后都必需有一printf的参数与之相呼应(只有当%%转换字符出现时会直接输出%字符)

    格式转换字符详解 “%[符号][宽度][.精度]类型”

    [宽度]:表示最少输出的字符个数

    [符号]:“-”表示对齐方式

    (1)%-8,左对齐,当显示字符不足8时,右补空格

    (2)%08,右对齐,当显示字符不足8时,左补0

    [.精度]对于浮点数表示小数点后的位数

    数值小数点后的位数大于显示精度,则只能显示[.精度]个小树位数(四舍五入),如果数值小数点后的位数小于显示精度,则补零。

    %.5 小数点后显示5位

    类型

    (1)%d:用于显示十进制有符号数,char,short,int,long long

    (2)%u:用于显示十进制无符号数,unsinged short,unsigned int,

    unsigned long long

    (3)%x: 用于显示十六进制整数,所有有符号及无符号整型

    (4)%f:用于显示十进制浮点数,float,double

    (5)%c:显示字符

    (6)%s:显示字符串

    printf(“%s”,xxx)puts(xxx)的区别:puts函数会自动添加换行,而printf(“%s”,……)不会。

    5. scanf的使用

    定义函数 int scanf(const char * format,...);

    函数说明 scanf()会将输入的数据根据参数format字符串来转换并格式化数据。Scanf()格式转换的一般形式如下:

    “%[宽度][数据所占字节数]输入类型”

    [宽度]:最多输入的字符个数

    [数据类型]:

    h表示两字节,short 

    l表示八字节,用于long long和 double

    什么都没有表示四字节

    [数据类型]输入类型

    (1)%d:int 

    (2)%f:float 

    (3)%lf:double 

    (4)%hd:short 

    scanf(“%c”,&x) 等价于 x = getchar(),虽然getchar的返回值是int类型,但不影响使用

    (5)%s:字符串

    用scanf读取字符串时,忽略前导的空白符,再次遇到空白符会结束输入,并将再次遇到的空白符留在缓冲区内,自动添加字符串数组的结束标志'n'

    1
    2
    3
    4
    5
    6
    7
    8
    #include <stdio.h>
    void main(int argc, char* argv[]){
        char a[20]; int ch; scanf_s("%s",a,20);
        printf("%s ", a);
        while ((ch = getchar()) != EOF ){
             putchar(ch);
        }
    }

    clip_image004

    我们输入i love you(ctrl+z)

    scanf_s 读取字符'i'以后结束(i后是空格),通过getchar函数第一个读取的字符就是空格,getchar会一直读取缓冲区中,直到缓冲区为空。

    6. fgets、fputs、fscanf、fprintf、fgetchar、fputchar

    上述方法只是多了个参数FILE * stream,表示这时的输入以stream指定的文件作为输入或者输出

    char* fgets(char* _Buf, int _MaxCount, FILE* _File);

    int fputs(const char * _Str,FILE* _File);

    int fprintf(FILE* _File, const char * _Format, ...);

    int fscanf(FILE* _File, const char * _Format, ...);

    int fgetc (FILE* _File) ;

    int fputc(int _Ch, FILE* _File);

    7. 其它相关函数

    int sprintf( char *_Dest, const char * format,...);

    函数说明sprintf和printf函数很类似,printf是将结果写入到标准输出流中,而sprintf是将结果写入到字符串数组_Dest中。返回值返回值返回参数str字符串长度,失败则返回-1。

    1
    2
    3
    4
    5
    6
    7
    #include<stdio.h>
    void main(){
        char* a = "This is string A!";
        char buf[80];
        sprintf_s(buf,"begin %s end ", a);
        printf("%s",buf);
    }

    clip_image006

    int sscanf_s(const char * _Src, const char * _Format, ...);

    sscanf函数与scanf类似,只不过scanf是从输入流中读取数据,而sscanf是从字符串数组_Src中读取数据

    1
    2
    3
    4
    5
    6
    7
    #include<stdio.h>
    void main(int argc, char* args[]){
        int i; double n;
        char str[20] = "123 3.1415";
        sscanf_s(str, "%d%lf", &i, &n);
        printf("%d %f ", i, n);
    }

    clip_image008

    设置流缓冲

    int fflush(FILE* stream);

    void setbuf(FILE* stream, char* buf);

    int setvbuf(FILE* stream, char* buf, int mode);

    数据总是先写入(或者读取)到流中,当缓冲区满了后,在将其写入到设备(或者获取读取到程序中),这样的工作方式效率更高。但是有时候我们可能需要更快的相应速度,我们可以调用fflush方法来冲刷缓冲区,注意这里冲刷的意思不是将缓冲区的内容删除,而是将还未满的缓冲区中的内容写入到设备(或者读取到程序中)

    setbuf中可以由参数buf自己设定缓冲区的位置和大小(大小由buf数组的大小决定)。

    setvbuf中的第三个参数决mode定了缓冲区的缓冲类型。它由三种取值

    _IOFBU:全满缓冲类型

    _IOLBU:行满缓冲类型

    _IONBU:无缓冲类型

  • 相关阅读:
    今天看到的一些好文语句摘录
    laravel5 怎么获取数组形式的数据
    laravel5.4将excel表格中的信息导入到数据库中
    phpmax_execution_time
    laravel 5.4 导出excel表格
    使用php函数ini_set()重新设置某个配置的设置值
    艾伟_转载:VS.NET 2010 新功能 [整理] 狼人:
    艾伟_转载:[一步一步MVC]第五回:让TagBuilder丰富你的HtmlHelper 狼人:
    艾伟_转载:.NET 4.0中数组的新增功能 狼人:
    艾伟_转载:一个简单的 Generic Factory 类 狼人:
  • 原文地址:https://www.cnblogs.com/feng9exe/p/8194862.html
Copyright © 2020-2023  润新知