與緩衝區相關的操作的常見的使我們的字元的輸入輸出操作,其實在接觸過作業系統和編譯原理之後,感覺電腦其實就是在處理一串又一串的字串。今天給大家介紹get家族的幾個函數。
很多最初使用C語言進行字串操作,並且認識到緩衝區問題的一般都是從類似下面的程式開始的:
很多初學者會問,按下斷行符號,為什麼沒等第二次輸入程式就結束了,這就是緩衝區的原因了。這裡需要瞭解一下getchar函數,這也是今天第一個要給大家介紹的stdio家族中getchar。
(1) int getchar(void)
函數每次從stdin緩衝區讀入一個字元遇到斷行符號返回,並且斷行符號也被存入緩衝區,所以getchar可以用來吃掉斷行符號符。
函數返回讀到的第一個字元的ASC碼值,如果失敗則返回-1
其定義為宏定義,即 #define getchar() getc(stdin)
因為是宏定義函數,所以getchar不支援指標的引用。
現在解釋跳過第二個輸入的原因,假如我們第一次輸入a按斷行符號,這時候我們getchar實際上將a字元和斷行符號放入了緩衝區中,並返回第一個字元的值,然後第二個getchar函數直接從緩衝區繼續去字元,拿到了斷行符號,沒有需要使用者再輸入。為什麼會有這種機制,可以參考
http://www.cnblogs.com/octobershiner/archive/2011/12/06/2278492.html
所以很多人把getchar用來吃掉斷行符號符,起到類似清空緩衝區的原因。清空緩衝區還可以使用fflush函數,但是這個函數不是C標準庫中的函數,有時候是無效的。
剛才說到了getchar的定義,其實就是getc函數的一種特殊的情況,下面介紹getc函數。
(2) int getc(FILE* stream)
也是採用的宏定義,所以不支援函數指標調用。
#define getc(_stream) (--(_stream)->_cnt >= 0 ? 0xff & *(_stream)->_ptr++ : _filbuf(_stream))
getc從指定的流中讀取一個字元,剛才的getchar實際上就是getc(stdin) ,stdin是標準輸入資料流,C在stdio.h中定義了三個流,也可以理解為是緩衝區。
#define stdin (&_iob[0]) //標準輸入,一般指向鍵盤
#define stdout (&_iob[1]) //標準輸出
#define stderr (&_iob[2]) //錯誤流
基本實現過程
在這裡再補充兩個函數,簡單帶過,因為getch和getche不是C標準的庫函數。
(3) int getch(void)
從命令列讀取一個字元,不顯示在命令列,很多人用來類比“按任意鍵繼續的效果”
(4) int getche(void)
從命令列中讀取一個字元,顯示在,命令列,與getch一樣不是標準C函數庫中的函數,在Windows平台下包含與conio.h中,在linux下與之相對的是curses.h庫,但是在cygwin類比的UNIX和ubuntu linux下gcc編譯器均無法使用。
本文的最後介紹一下gets函數,他從流中讀取字串直到遇到分行符號或者EOF或者遇到錯誤.
(5) char* gets(char* buffer)
從stdin流中讀取字串,直至接受到分行符號或EOF時停止,並將讀取的結果存放在buffer指標所指向的字元數組中。分行符號不作為讀取串的內容,讀取的分行符號被轉換為null值,並由此來結束字串。
其實gets這個函數還是有很多安全性的問題需要注意的,比如他要一直等EOF或者分行符號才會結束,所以當程式無法預知可能存在的輸入串長度時,是非常危險的,上一篇文章提到過緩衝區溢位的攻擊,那麼這裡就是一個漏洞,我們不可能和攻擊者比buffer的大小,及時我們定義很大很大的緩衝區,攻擊者卻能輸入一個更長的字串。
其中VS 2005中提供了一個新的函數gets_s函數,帶有安全性的gets,但是顯然這不是標準庫中所定義的。
========f的分割線=================
stdio.h中還包含了與gets和getc想對的檔案操作函數,fgets和fgetc
char *fgets(char *s, int n, FILE *stream); // n表示一次讀入資料的長度,從stream中讀取存入到s串中,遇到換行和結束符則結束
int fgetc(FILE * stream);
後續會分享put函數的系列。。。。