1. 程式人生 > >格式化io與非格式化io得對比理解

格式化io與非格式化io得對比理解

測試函式1

#include<stdio.h>
#include<string.h>
int main()
{
    int a[5]={65,66,67,68};
    char value[10],i;
    FILE *pf;

    if(!(pf = fopen("printf.txt","w+")))
    {
        perror("open file:");
        return -1;
    }

    fprintf(pf,"%d %c %d %c",a[0],a[1],a[2],a[3]);
    sprintf(value,"%d
%c %d %c"
,a[0],a[1],a[2],a[3]); puts(value); printf("%d",strlen(value)); fclose(pf); return 0; }

這個程式的輸出是:
65 B 67 D
9
從這個程式發現scanf家族對這個函式所作的事情就是將所有格式的變數以
一個文字的形式輸出到我們指定的地方,不論是陣列還是檔案;

sprintf(value,”%d %c %d %c”,a[0],a[1],a[2],a[3]);
這個呼叫,
即先讀到一個%d,對應a[0]是65,
就在value[0]放入6的ascii碼然後在value[1]存入5的ascii碼,下一個是空格就在value[2]儲存空格的ascii
下一個是%c匹配a[2]就把a[2]的值直接當作一個ascii碼存入value[3],以此類推….
所以最後得到字串長度為9

測試程式2

#include<stdio.h>

int main()
{
    FILE *fp;

    int a;
    char b,c[100];

    if(!(fp = fopen("scanf.txt","r+")))
    {
        perror("opend file:");
        return -1;
    }

    fscanf(fp,"%d %c %s",&a,&b,c);
    printf("%d %d %d\n",(int)a,(int)b,(int)*c);

    fclose(fp);

    return
0; }

scanf.txt的內容是:
123 a abc
輸出是:
123 97 97

scanf家族的原理也和printf相似:第一個引數型別是%d即讀入一個整形賦值給a;
用一個空格分開123和a
就是為了讓程式瞭解空格前1,2,3這三個字元都屬於變數a;

所以a的值是經過:
(‘1’-‘0’) * 100 * +(‘2’-‘0’) * 10 + (‘3’-‘0’) * 1得到的;

得出結論:

理解二進位制儲存的方式和文字儲存的方式:
在linux下:所有的檔案都是按文字的方式儲存和讀取的,所以所有的檔案開啟的時候預設為是文字檔案,當我們儲存一些數字進入文件時,只能按位元組翻譯數字為對應的符號,這就是為什麼開啟一個可執行檔案時會出現亂碼的情況.因為可執行檔案都是一些二進位制機器碼.

檔案儲存是一個個位元組儲存的,而且儲存的都是一個二進位制數,scanf和print只是用了字元數字互轉的方法對檔案或者陣列元素
進行取放的.

對於格式化輸入輸出函式

檔案儲存是一個個位元組儲存的,而且儲存的都是一個二進位制數,scanf和print只是用了字元數字互轉的方法對檔案或者陣列元素
進行取放的.

即:scanf按格式把 ascii碼->所需格式儲存至變數.
printf按格式把 某些格式變數->ascii碼
方便文字軟體按ascii碼顯示內容

至於非格式化io,即不格式化直接將資料原封不動一個個位元組輸出輸入;

fgetc和getchar;fputc和putchar:即一個個位元組輸入輸出;
fgets和gets;fputs和puts:即一次一行’\n’一個個位元組輸入輸出;
fread和fwrite:即一次可控位元組數輸入輸出;(直接忽略變數型別直接對記憶體按位元組操作)