系統調用IO和標準IO

来源:https://www.cnblogs.com/songhe364826110/archive/2019/09/18/11546076.html
-Advertisement-
Play Games

[TOC] 1. 系統調用IO(無緩衝IO) 系統調用 在Linux中一切皆文件,文件操作在Linux中是十分重要的。為此, Linux內核提供了一組用戶進程與內核進行交互的介面用於對文件和設備進行訪問控制,這些介面被稱為系統調用。 系統調用對於應用程式最大的作用在於: 以統一的形式,為應用程式提供 ...


目錄

1. 系統調用IO(無緩衝IO)

系統調用

在Linux中一切皆文件,文件操作在Linux中是十分重要的。為此, Linux內核提供了一組用戶進程與內核進行交互的介面用於對文件和設備進行訪問控制,這些介面被稱為系統調用。

系統調用對於應用程式最大的作用在於:

  • 以統一的形式,為應用程式提供了一組文件訪問的抽象介面,應用程式不需要關心文件的具體類型,也不用關心其內部實現細節。

常用系統調用IO函數

常用的系統調用IO函數有5個:open、close、read、write、ioctl,此外還有個非系統調用IO函數lseek,系統調用IO都是不帶緩衝的IO。

open

open用於創建一個新文件或打開一個已有文件,返回一個非負的文件描述符fd。
0、1、2為系統預定義的文件描述符,分別代表STDIN_FILENO、STDOUT_FILENO、STDERR_FILENO。

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

//成功返迴文件描述符,失敗返回-1
int open(const char *pathname, int flags, ... /* mode_t mode */);

flags參數一般在O_RDONLY、O_WRONLY和O_RDWR中選擇指定一個,還可以根據需要或上以下常值:

  • O_CREAT:若文件不存在則創建它,此時需要第三個參數mode,mode可設的值及含義如下圖所示。
  • O_APPEND:每次寫時都追加到文件的尾端
  • O_NONBLOCK:如果pathname對應的是FIFO、塊特殊文件或字元特殊文件,則該命令使open操作及後續IO操作設定為非阻塞模式

close

close用於關閉一個已打開文件。

#include <unistd.h>

//成功返回0,失敗返回-1
int close(int fd);

進程終止時,內核會自動關閉它所有的打開文件,應用程式經常利用這一點而不顯式關閉文件。

read

read用於從打開文件中讀數據。

#include <unistd.h>

//成功返回讀到的位元組數;若讀到文件尾則返回0;失敗返回-1
ssize_t read(int fd, void *buf, size_t count);

read操作從文件的當前偏移量處開始,在成功返回之前,文件偏移量將增加實際讀到的位元組數。
有幾種情況可能導致實際讀到的位元組數少於要求讀的位元組數:

  • 讀普通文件時,在讀到要求位元組數之前就到達了文件尾。例如,離文件尾還有30位元組,要求讀100位元組,則read返回30,下次在調用read時會直接返回0
  • 從網路讀時,網路中的緩衝機制可能造成返回值少於要求讀的位元組數,解決辦法在網路編程專題中再講

write

write用於向文件寫入數據。

#include <unistd.h>

//成功返回寫入的位元組數,失敗返回-1
ssize_t write(int fd, const void *buf, size_t count);
  • write的返回值通常與參數count相同,否則表示出錯。
  • 對於普通文件,write操作從文件的當前偏移量處開始
  • 若指定了O_APPEND選項,則每次寫之前先將文件偏移量設置到文件尾
  • 成功寫入之後,文件偏移量增加實際寫的位元組數。

lseek

lseek用於設置打開文件的偏移量。

#include <sys/types.h>
#include <unistd.h>

//成功返回新的文件偏移量,失敗返回-1
off_t lseek(int fd, off_t offset, int whence);

對offset的解釋取決於whence的值:

  • 若whence == SEEK_SET,則將文件偏移量設為距文件開頭offset個位元組,此時offset必須為非負
  • 若whence == SEEK_CUR,則將文件偏移量設為當前值 + offset,此時offset可正可負
  • 若whence == SEEK_END,則將文件偏移量設為文件長度 + offset,此時offset可正可負

註意:

  • lseek僅將新的文件偏移量記錄在內核中,它並不引起任何IO操作,因此它不是系統調用IO,但該偏移量會用於下一次read/write操作
  • 管道、FIFO和套接字不支持設置文件偏移量,不能對其調用lseek

ioctl

ioctl提供了一個用於控制設備及其描述符行為和配置底層服務的介面。

#include <sys/ioctl.h>

//出錯返回-1,成功返回其他值
int ioctl(int fd, int cmd, ...);
  • ioctl對描述符fd引用的對象執行由cmd參數指定的操作
  • 每個設備驅動程式都可以定義它自己專用的一組ioctl命令

2. 標準IO(帶緩衝IO)

概述

標準IO其實就是stdio.h頭文件中提供的IO介面,只不過在特定的系統中可能有特定的內部實現。
和系統調用IO類似,標準IO也預定義了三個文件指針stdin、stdout、stderr,分別對應標準輸入、標準輸出、標準錯誤。

緩衝與沖洗

標準IO是帶緩衝的IO,一共有3種類型的緩衝:

  • 全緩衝:緩衝區填滿後才進行IO操作,如磁碟文件
  • 行緩衝:遇到換行符才進行IO操作,如命令行終端(stdin和stdout)
  • 無緩衝:不經過緩衝,立即進行IO操作,如stderr

一般情況下,系統預設使用下列類型的緩衝:

  • stderr是無緩衝的
  • 指向終端設備的流是行緩衝的,否則是全緩衝的

對於一個打開的流,可以調用setbuf或setvbuf改變其緩衝類型.

//成功返回0,失敗返回非0
void setbuf(FILE *fp, char *buf);
int setvbuf(FILE *fp, char *buf, int mode, size_t size);
  • setbuf用於關閉或打開fp的緩衝機制,若打開,則buf必須指向一個長度為BUFSIZ的緩衝區;若關閉,則buf設為NULL
  • setvbuf通過參數mode可精確設置緩衝類型,_IOLBF==全緩衝, _IOLBF==行緩衝,_IONBF==無緩衝
  • 當setvbuf設置為帶緩衝時,buf必須指向一個長度為size的緩衝區,推薦將buf設為NULL讓系統自動分配緩衝區長度,此時size可設為0

對於全緩衝和行緩衝,不管是否滿足IO條件,都可以使用fflush函數強制進行IO操作,我們稱其為沖洗。

//成功返回0,失敗返回EOF
//若fp為NULL,將導致沖洗所有輸出流
int fflush(FILE *fp);

常用標準IO函數

常用的標準IO函數分為以下幾大類:

  • 打開和關閉流
  • 定位流
  • 讀寫流,包括文本IO、二進位IO和格式化IO三種

打開和關閉流

//成功返迴文件指針,失敗返回NULL
FILE *fopen(const char *pathname, const char *type);

//成功返回0,失敗返回EOF
void fclose(FILE *fp);

fopen打開由pathname指定的文件,type指定讀寫方式:

  • 使用字元b作為type的一部分,使得標準IO可以區分文本文件和二進位文件
  • Linux內核不區分文本和二進位文件,因此在Linux系統下使用字元b實際上沒有作用,只讀、只寫、讀寫分別指定為"r"、"w"、"rw"即可
  • Windows中,文本文件只讀、只寫、讀寫分別為"r"、"w"、"rw",二進位文件只讀、只寫、讀寫分別為"rb"、"wb"、"rb+"
  • 只讀方式要求文件必須存在,只寫或讀寫方式會在文件不存在時創建

fclose關閉文件,關閉前緩衝區中的輸出數據會被沖洗,輸出數據則丟棄。

定位流

流的定位類似於系統調用IO中獲取當前文件偏移量,ftell和fseek函數可用於定位流。

//成功返當前文件位置,出錯返回-1
int ftell(FILE *fp);

//成功返回0,失敗返回-1
void fseek(FILE *fp, long offset, int whence);

offset和whence含義及可設的值與系統調用IO中的lseek相同,不再贅述,但如果是在非Linux系統,則有一點需要註意:

  • 對於二進位文件,文件位置嚴格按照位元組偏移量計算,但對於文本文件可能並非如此
  • 定位文本文件時,whence必須是SEEK_SET,且offset只能是0或ftell返回值

文本IO

文本IO有兩種:

  • 每次讀寫一個字元
  • 每次讀寫一行字元串
/*
 * 每次讀寫一個字元
*/

//成功返回下一個字元,到達文件尾或失敗返回EOF
int getc(FILE *fp);          //可能實現為巨集,因此不允許將其地址作為參數傳給另一個函數,因為巨集沒有地址
int fgetc(FILE *fp);         //一定是函數
int getchar();               //等價於getc(stdin)

//成功返回c,失敗返回EOF
int putc(int c, FILE *fp);   //可能實現為巨集,因此不允許將其地址作為參數傳給另一個函數,因為巨集沒有地址
int fputc(int c, FILE *fp);  //一定是函數
int putchar(int c);          //等價於putc(c, stdout)
/*
 * 每次讀寫一行字元串
*/

//成功返回str,到達文件尾或失敗返回EOF
char *fgets(char *str, int n, FILE *fp); //從fp讀取直到換行符(換行符也讀入),str必須以'\0'結尾,故包括換行符在內不能超過n-1個字元

//成功返回非負值,失敗返回EOF
int fputs(const char *str, FILE *fp);    //將字元串str輸出到fp,str只要求以'\0'結尾,不一定含有換行符

二進位IO

二進位IO就是fread和fwrite。

//返回讀或寫的對象數
size_t fread(void *ptr, size_t size, size_t nobj, FILE *fp);
size_t fwrite(const void *ptr, size_t size, size_t nobj, FILE *fp);

二進位IO常見的用法包括:

  • 讀寫一個二進位數組
  • 讀寫一個結構

上述兩種用法結合起來,還可以實現讀寫一個結構數組。

struct Item
{
    int id;
    char text[100];
};

int data[10];
struct Item item;
struct Item item[10];

//讀寫二進位數組
fread(&data[2], sizeof(int), 4, fp);
fwrite(&data[2], sizeof(int), 4, fp);

//讀寫結構
fread(&item, sizeof(item), 1, fp);
fwrite(&item, sizeof(item), 1, fp);

//讀寫結構數組
fread(&item, sizeof(item[0]), sizeof(item) / sizeof(item[0]), fp);
fwrite(&item, sizeof(item[0]), sizeof(item) / sizeof(item[0]), fp);

格式化IO

格式化IO包括輸入函數族和輸出函數族,我們剔除了不常用的與文件指針fp、文件描述符fd相關的API,僅保留常用的3個輸出函數和2個輸入函數。

//成功返回輸出或存入buf的字元數(不含'\0'),失敗返回負值
int printf(const char *format, ...);
int sprintf(char *buf, const char *format, ...);
int snprintf(char *buf, size_t n, const char *format, ...);
  • sprintf和snprintf會自動在buf末尾添加字元串結束符'\0',但該字元不包括在返回值中
  • snprintf要求調用者保證緩衝區buf長度n足夠大
//成功返回輸入的字元數,到達文件尾或失敗返回EOF
int scanf(const char *format, ...);
int sscanf(const char *buf, const char *format, ...);

PS:sscanf在實際工程中有一個實用的小技巧:串口接收的一條報文,可以根據串口協議,使用sscanf提取各個欄位,從而快速便捷的進行報文解析。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 屬性(Property)是類(class)、結構(structure)和介面(interface)的命名(named)成員。類或結構中的成員變數或方法稱為 域(Field)。屬性(Property)是域(Field)的擴展,且可使用相同的語法來訪問。它們使用 訪問器(accessors) 讓私有域的 ...
  • 場景 Winform中使用FastReport實現簡單的自定義PDF導出: https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/details/100920681 在上面已經實現自定義PDF並導出的基礎上,實現導出或者列印前的預覽。 實現 在窗體中拖拽 ...
  • 下拉刷新: html: js: 無限滾動: html: js: 解釋: 如有問題,歡迎留言 ...
  • 目錄導航 一、安裝 二、配置 三、調用 四、錯誤記錄 一、安裝 新建一個沒有身份驗證的mvc項目 - SwaggerMvc5Demo,然後添加一個名為Remote(自定義)且包含基礎讀寫(不想手寫)的ApiController 開源地址:https://github.com/domaindriven ...
  • 前言 不多BB,直接上圖 Linux命令行的組成結構 Linux系統命令操作語法格式 | 命令 | 空格 | 參數 | 空格 | 【文件或路徑】需要處理的內容 | | | | | | | | rm | | rf | | /tmp/ | | ls | | la | | /home | | 結婚 | | ...
  • [TOC] 由Linux內核提供的基本時間是自1970 01 01 00:00:00 +0000 (UTC)這一特定時間以來經過的秒數,這種描述是以數據類型time_t表示的,我們稱其為日曆時間。 獲得日曆時間的函數有3個:time、clock_gettime和gettimeofday。 time函 ...
  • [TOC] 1. 文件類型 Linux下一切皆文件,但文件也分為7種類型,文件類型信息包含在struct stat結構的st_mode成員中,可以用下表中的巨集來檢測文件類型,這些巨集的參數都是stat.st_mode。 | 文件類型 | 說 明 | 檢測文件類型的巨集 | | | | | 普通文件 | ...
  • [TOC] 1. IO模型 Linux下可用的IO模型有5種,分別是: 阻塞式IO 非阻塞式IO IO復用 信號驅動式IO(SIGIO) 非同步IO(Posix的aio_系列函數) 其中,除了非同步IO,其餘都屬於同步IO模型。 在這5種模型中,我們目前只關註前3種,並且把IO復用放在網路編程專題中講, ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...