格式化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:即一次可控位元組數輸入輸出;(直接忽略變數型別直接對記憶體按位元組操作)