linux學習筆記8-linux系統io開發知識sets

weixin_34138377發表於2018-03-04

資源和工具

基礎知識

工程示例

x.1 read/write

read/write

read函式從開啟的裝置或檔案中讀取資料。
#include <unistd.h>  ssize_t read(int fd, void *buf, size_t count); 返回值:成功返回讀取的位元組數,出錯返回-1並設定errno,如果在調read之前已到達檔案末尾,則這次read返回0
引數count是請求讀取的位元組數,讀上來的資料儲存在緩衝區buf中,同時檔案的當前讀寫位置向後移。注意這個讀寫位置和使用C標準I/O庫時的讀寫位置有可能不同,這個讀寫位置是記在核心中的,而使用C標準I/O庫時的讀寫位置是使用者空間I/O緩衝區中的位置。

fread就是通過read來實現的,fread是C語言的庫,而read是系統呼叫
但是差別在read每次讀的資料是呼叫者要求的大小,比如呼叫要求讀取10個位元組資料,read就會讀10個位元組資料到陣列中,而fread不一樣,為了加快讀的速度,fread每次都會讀比要求更多的資料,然後放到緩衝區中,這樣下次再讀資料只需要到緩衝區中去取就可以了。

fread每次會讀取一個緩衝區大小的資料,32位下一般是4096個位元組,相當於呼叫了read(fd,buf,4096)

比如需要讀取512個位元組資料,分4次讀取,呼叫read就是:
for(i=0; i<4; ++i)
read(fd,buf,128)
一共有4次系統呼叫

而fread一次就讀取了4096位元組放到緩衝區了,所以省事了


比如用fgetc讀一個位元組,fgetc有可能從核心中預讀1024個位元組到I/O緩衝區中,再返回第一個位元組,這時該檔案在核心中記錄的讀寫位置是1024,而在FILE結構體中記錄的讀寫位置是1。注意返回值型別是ssize_t,表示有符號的size_t,這樣既可以返回正的位元組數、0(表示到達檔案末尾)也可以返回負值-1(表示出錯)。read函式返回時,返回值說明了buf中前多少個位元組是剛讀上來的。有些情況下,實際讀到的位元組數(返回值)會小於請求讀的位元組數count,例如:
讀常規檔案時,在讀到count個位元組之前已到達檔案末尾。例如,距檔案末尾還有30個位元組而請求讀100個位元組,則read返回30,下次read將返回0。

從終端裝置讀,通常以行為單位,讀到換行符就返回了。

從網路讀,根據不同的傳輸層協議和核心快取機制,返回值可能小於請求的位元組數,後面socket程式設計部分會詳細講解。

相關文章