a一级爱做片免费观看欧美,久久国产一区二区,日本一二三区免费,久草视频手机在线观看

博客專欄

EEPW首頁 > 博客 > 嵌入式Linux:文件I/O和標準I/O庫

嵌入式Linux:文件I/O和標準I/O庫

發布人:美男子玩編程 時間:2024-05-30 來源:工程師 發布文章

文件 I/O (Input/Output)和標準 I/O 庫是用于在 C 語言中進行文件操作的兩種不同的方法。

1文件I/O

文 I/O(Input/Output)是指程序與文件之間進行數據交換的過程。在計算機編程中,文件 I/O 是通過讀取和寫入文件來實現數據的輸入和輸出操作。文件 I/O 主要涉及打開文件、讀取文件內容、寫入文件內容和關閉文件等操作。


常見文件 I/O 操作包括使用系統調用(如 open()、read()、write()、close())來進行文件操作。通過文件 I/O,程序可以從文件中讀取數據,對數據進行處理,然后將結果寫入文件中,實現數據的持久化存儲和處理。


在Linux系統中,一切皆文件是其核心設計理念之一,因此文件I/O操作在Linux系統中顯得尤為重要。


1.1、文件描述符


文件描述符是操作系統中用于標識打開文件的整數值。它是進程與文件之間的橋梁,允許進程對文件進行讀取、寫入和其他操作。在Linux系統中,每個打開的文件都與一個文件描述符相關聯,這個文件描述符是一個非負整數,通常是從0開始遞增的。


文件描述符直接與操作系統的文件表項相關聯,是操作系統提供的抽象。


舉例來說,假設我們有一個C語言程序,打開了一個名為“example.txt”的文本文件進行讀取。在這個程序中,文件描述符是用于表示這個打開的文件的整數值。當程序調用open函數打開文件時,操作系統會分配一個文件描述符,并將其返回給程序。程序可以使用這個文件描述符執行讀取操作,如讀取文件內容并將其輸出到終端上。


#include#include#include

int main() {    int fd; // 文件描述符    char buf[1024]; // 用于存儲讀取的數據
   // 打開文件 example.txt    fd = open("example.txt", O_RDONLY);    if (fd == -1) {        perror("open");        return 1;    }
   // 讀取文件內容并輸出到終端上    ssize_t bytes_read;    while ((bytes_read = read(fd, buf, sizeof(buf))) > 0) {        write(STDOUT_FILENO, buf, bytes_read);    }
   // 關閉文件    close(fd);
   return 0;}


在這個示例中,open函數打開文件example.txt并返回一個文件描述符,然后read函數使用這個文件描述符來從文件中讀取數據。最后,close函數關閉文件,并釋放對應的文件描述符。


1.2、open打開文件

在Linux系統中,操作文件需要先打開它以獲取文件描述符,然后進行讀寫或其他操作,最后關閉文件。open函數可用于打開現有文件或創建新文件。函數原型如下所示:


#include#include#include
int open(const char *pathname, int flags);int open(const char *pathname, int flags, mode_t mode);


函數的參數和返回值含義如下:


  • pathname:字符串類型,用于標識需要打開或創建的文件。它可以包含路徑信息,可以是絕對路徑或相對路徑,例如:"./src_file"(當前目錄下的 src_file 文件)或 "/home/dengtao/hello.c"。如果 pathname 是一個符號鏈接,open 函數會對其進行解引用。

  • flags:調用 open 函數時需要提供的標志,包括文件訪問模式標志以及其他文件相關標志。這些標志使用宏定義進行描述,并都是常量。open 函數提供了豐富的標志選項,我們可以單獨使用某一個標志,也可以通過位或運算(|)將多個標志進行組合。

  • mode:用于指定新建文件的訪問權限,僅在flags參數中包含O_CREAT或O_TMPFILE標志時有效。在Linux系統中,權限對于文件是一個重要的屬性。我們可以使用touch命令在Linux系統中創建一個文件,此時文件會有默認的權限。如果需要修改文件權限,可以使用chmod命令進行修改。例如,在Linux系統下,我們可以使用ls -l命令查看文件對應的權限。

  • 返回值:成功將返回文件描述符,文件描述符是一個非負整數;失敗將返回-1。


open函數的flags參數用于指定打開文件時的行為和權限。下面是一些常用的flags參數值:


  • O_RDONLY:只讀方式打開文件。

  • O_WRONLY:只寫方式打開文件。

  • O_RDWR:讀寫方式打開文件。

  • O_CREAT:如果文件不存在,則創建文件。

  • O_EXCL:與O_CREAT一同使用,如果文件已經存在,則返回錯誤。

  • O_TRUNC:如果文件存在且為只寫或讀寫打開,則將其長度截斷為0。

  • O_APPEND:追加方式打開文件,在寫入數據時追加到文件末尾。

  • O_NONBLOCK:非阻塞方式打開文件,在沒有數據可讀取時不阻塞。

  • O_SYNC:同步寫入方式打開文件,對寫入文件的每個操作進行同步。

  • O_DIRECT:直接IO方式打開文件,繞過系統緩存,數據直接讀寫到磁盤。

  • O_TMPFILE:創建一個臨時文件,文件在關閉時自動刪除。


open函數的常用的mode參數:


  • S_IRUSR:文件所有者讀權限。

  • S_IWUSR:文件所有者寫權限。

  • S_IXUSR:文件所有者執行權限。

  • S_IRGRP:文件組用戶讀權限。

  • S_IWGRP:文件組用戶寫權限。

  • S_IXGRP:文件組用戶執行權限。

  • S_IROTH:其他用戶讀權限。

  • S_IWOTH:其他用戶寫權限。

  • S_IXOTH:其他用戶執行權限。


在應用程序中使用 open 函數時,需要包含 3 個頭文件“#include”、“#include”、“#include”。


下面是一個使用 open 函數的簡單示例:


#include#include#include#include
int main() {    const char *filename = "example.txt";    int fd;
   // 使用 open 函數打開文件,如果文件不存在,則創建    fd = open(filename, O_WRONLY | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR);    if (fd == -1) {        perror("open");        exit(EXIT_FAILURE);    }
   // 向文件寫入內容    if (write(fd, "Hello, World!", 13) == -1) {        perror("write");        close(fd);        exit(EXIT_FAILURE);    }
   // 關閉文件    if (close(fd) == -1) {        perror("close");        exit(EXIT_FAILURE);    }
   printf("File created and written successfully.n");
   return 0;}


在這個示例中:


  • 我們首先定義了一個文件名 example.txt。

  • 使用 open 函數打開文件,使用 O_WRONLY 標志表示以只寫方式打開文件,O_CREAT 標志表示如果文件不存在則創建,O_TRUNC 標志表示如果文件存在則將其截斷為空文件,最后一個參數 S_IRUSR | S_IWUSR 指定了新創建文件的權限為用戶可讀可寫。

  • 如果 open 函數調用失敗,會打印錯誤消息并退出程序。

  • 使用 write 函數向文件中寫入內容。

  • 最后使用 close 函數關閉文件。


1.3、write寫文件


write 函數用于將數據寫入文件。其函數原型如下:


#include
ssize_t write(int fd, const void *buf, size_t count);


函數的參數和返回值含義如下:


  • fd:文件描述符,代表要寫入數據的文件。需要將要寫入數據的文件對應的文件描述符傳遞給 write 函數。

  • buf:指定要寫入數據的緩沖區。

  • count:指定要寫入的字節數。

  • 返回值:成功時返回寫入的字節數(0 表示未寫入任何字節)。如果返回值小于 count 參數,這不一定是錯誤,例如磁盤空間已滿可能導致未寫入所有字節。如果寫入出錯,則返回 -1。



對于普通文件,無論是讀取還是寫入,一個關鍵問題是確定從文件的哪個位置開始進行操作。即所謂的I/O操作位置偏移量。讀寫操作都從文件的當前位置偏移量開始。默認情況下,當前位置偏移量通常是0,即指向文件的起始位置。隨著read、write函數的調用,當前位置偏移量也會相應移動。例如,如果當前位置偏移量為1000字節,調用write()寫入或read()讀取500字節后,當前位置偏移量將移動到1500字節處。


使用 write 函數需要先包含 unistd.h 頭文件。


下面是一個示例代碼,將字符串寫入文件:


#include#include#include
int main() {    int fd = open("example.txt", O_WRONLY | O_CREAT, 0644);    if (fd == -1) {        perror("open");        return 1;    }
   const char *message = "Hello, world!";    ssize_t bytes_written = write(fd, message, strlen(message));    if (bytes_written == -1) {        perror("write");        close(fd);        return 1;    }
   close(fd);    printf("Data written successfully.n");    return 0;}


在此示例中,我們首先打開了一個文件 example.txt 以供寫入,然后使用 write 函數將字符串 "Hello, world!" 寫入文件中。 


1.4、read讀文件

調用 read 函數可從打開的文件中讀取數據,其函數原型如下所示:

#include
ssize_t read(int fd, void *buf, size_t count);


函數參數和返回值含義如下:

  • fd:文件描述符,用于標識要讀取的文件。

  • buf:用于存儲讀取數據的緩沖區。

  • count:需要讀取的字節數。

  • 返回值:如果讀取成功,返回讀取到的字節數。實際讀取到的字節數可能小于請求的字節數,也可能為0,例如當文件已到達末尾時。



使用 read 函數需要先包含 unistd.h 頭文件。


例如,下面是一個簡單的示例,從文件中讀取數據:


#include#include#include#include
#define BUF_SIZE 1024
int main() {    int fd;    ssize_t bytes_read;    char buffer[BUF_SIZE];
   // 打開文件    fd = open("example.txt", O_RDONLY);    if (fd == -1) {        perror("open");        exit(EXIT_FAILURE);    }
   // 讀取文件內容    bytes_read = read(fd, buffer, BUF_SIZE);    if (bytes_read == -1) {        perror("read");        exit(EXIT_FAILURE);    }
   // 輸出讀取的內容    write(STDOUT_FILENO, buffer, bytes_read);
   // 關閉文件    close(fd);
   return 0;}



這個示例打開一個名為example.txt的文件,從中讀取數據并將其寫入標準輸出。



1.5、close關閉文件


close 函數用于關閉一個已經打開的文件描述符,釋放對應的資源。在Linux系統中,文件描述符是有限資源,因此在不再需要使用文件時,應該及時關閉,以釋放資源并避免資源泄漏。


函數原型如下所示:


#include
int close(int fd);


函數參數和返回值含義如下:


  • fd:文件描述符,需要關閉的文件所對應的文件描述符。

  • 返回值:如果成功返回 0,如果失敗則返回-1。

使用 close 函數需要先包含頭文件。


以下是一個簡單的示例,演示如何使用 close 函數關閉文件:


#include#include#include#include
int main() {    // 打開一個文件,獲取文件描述符    int fd = open("example.txt", O_RDONLY);    if (fd == -1) {        perror("open");        exit(EXIT_FAILURE);    }
   // 關閉文件    if (close(fd) == -1) {        perror("close");        exit(EXIT_FAILURE);    }
   printf("File closed successfully.n");
   return 0;}



在這個例子中,首先通過 open 函數打開了一個文件,然后使用 close 函數關閉了文件描述符。perror 函數用于打印出發生錯誤的詳細信息。


除了使用 close 函數顯式關閉文件之外,在 Linux 系統中,當一個進程終止時,內核會自動關閉它打開的所有文件。這意味著如果一個程序在退出時沒有關閉已打開的文件,內核會代為關閉這些文件。許多程序都依賴于這一特性,因此沒有顯式地使用 close 函數來關閉文件。


然而,顯式關閉不再需要的文件描述符通常是良好的編程習慣。這樣做可以提高代碼的可讀性和可靠性,并確保在后續修改時代碼的行為符合預期。此外,釋放不再需要的文件描述符可以有效地管理有限的系統資源。


2


標準I/O庫


標準I/O庫是C語言中用于進行輸入和輸出操作的標準庫之一。它提供了一組函數和數據結構,用于與文件、終端設備、管道等進行交互,使得程序可以方便地進行輸入和輸出操作,而無需直接操作文件描述符。


標準I/O庫函數構建在文件I/O系統調用(如 open()、read()、write()、lseek()、close() 等)之上。例如,fopen() 利用 open() 系統調用打開文件,fread() 利用 read() 系統調用讀取文件,fwrite() 利用 write() 系統調用寫入文件等。


盡管標準I/O和文件I/O都是C語言函數,但它們有明顯區別:


  • 標準 I/O 是標準 C 庫函數,而文件 I/O 是 Linux 系統調用;

  • 標準 I/O 是文件 I/O 的封裝,實際上調用文件 I/O 完成操作;

  • 可移植性方面,標準 I/O 更優,因為不同操作系統的系統調用接口不同,而標準 I/O 接口幾乎相同;

  • 在性能和效率方面,標準 I/O 由于維護自己的緩沖區,性能更高,而文件 I/O 在用戶空間無緩存。


標準I/O庫通常包含在C標準庫中,其函數和數據結構被定義在頭文件中。一些常用的標準I/O函數包括fopen、fclose、fread、fwrite、fprintf、fscanf等。所以使用時候需要在程序源碼中包含頭文件。


標準I/O庫的主要特點包括:


  • 緩沖機制:標準I/O庫通常使用緩沖區來提高性能。例如,在輸出時,數據首先寫入到緩沖區,然后在適當的時機才會被刷新到實際的輸出設備上,從而減少了系統調用的次數,提高了效率。

  • 格式化輸入輸出:標準I/O庫提供了格式化輸入輸出的功能,例如printf和scanf函數允許以特定格式輸出和輸入數據,使得數據的處理更加方便。

  • 文件操作:標準I/O庫提供了一系列函數用于文件的打開、關閉、讀取、寫入等操作,例如fopen、fclose、fread、fwrite等。

  • 錯誤處理:標準I/O庫提供了一套錯誤處理機制,允許程序員檢測和處理輸入輸出操作中可能出現的錯誤情況。

使用標準I/O庫可以使得程序更加可移植,因為它們提供了對底層系統調用的封裝,使得程序不依賴于特定的操作系統或文件系統。因此,標準I/O庫是C語言中進行文件操作和輸入輸出的主要方式之一。


2.1、FILE指針

標準I/O庫函數操作圍繞FILE指針展開。調用標準I/O庫函數打開或創建文件時,返回一個指向FILE類型對象的指針(FILE *),該指針與被打開或創建的文件相關聯,用于后續的標準I/O操作。因此,FILE指針在標準I/O庫中扮演了與文件描述符類似的角色,但用于更高級別的操作。


FILE結構體包含了標準I/O庫函數所需的所有文件管理信息,如文件描述符、文件緩沖區指針、緩沖區長度、當前緩沖區字節數以及出錯標志等。


當使用標準I/O庫函數打開或創建文件時,會返回一個指向FILE類型對象的指針,該指針與被打開或創建的文件相關聯。下面是一個簡單的示例:


#include
int main() {    FILE *file_ptr;
   // 使用標準I/O庫函數打開文件    file_ptr = fopen("example.txt", "w");    if (file_ptr == NULL) {        printf("Failed to open file.n");        return 1;    }
   // 使用 FILE 指針進行寫入操作    fprintf(file_ptr, "Hello, world!n");
   // 關閉文件    fclose(file_ptr);
   return 0;}

在這個示例中,file_ptr指針與文件相關聯,用于后續的標準I/O操作(寫入操作)。通過FILE指針,我們可以方便地進行文件的讀寫操作,而不必直接操作文件描述符和底層的文件系統。


2.2、fopen打開文件

fopen() 是C語言標準庫中用于打開文件的函數之一。它的原型如下:




FILE *fopen(const char *filename, const char *mode);



函數參數和返回值含義如下:


  • path:參數 path 是一個指向文件路徑的指針,可以是文件的絕對路徑或相對路徑。這個路徑指定了要打開或創建的文件的位置和名稱。

  • mode:參數 mode 是一個字符串,指定了對文件的讀寫權限。它描述了打開或創建文件時所需的操作類型。常見的模式包括:

    • "r":只讀模式,用于打開一個已存在的文本文件,文件必須存在。

    • "w":寫入模式,用于創建一個新的空文本文件,如果文件已存在,則刪除其內容。

    • "a":追加模式,用于打開一個文本文件以便寫入,如果文件不存在,則創建文件,文件指針被放在文件的末尾。

    • "r+":讀寫模式,用于打開一個文本文件用于讀取和寫入,文件必須存在。

    • "w+":讀寫模式,用于創建一個新的空文本文件用于讀取和寫入,如果文件已存在,則刪除其內容。

    • "a+":讀寫模式,用于打開一個文本文件用于讀取和寫入,如果文件不存在,則創建文件,文件指針被放在文件的末尾。

  • 返回值:函數調用成功時,返回一個指向 FILE 類型對象的指針(FILE *),該指針與打開或創建的文件相關聯。后續的標準 I/O 操作將圍繞這個 FILE 指針進行。如果函數調用失敗,則返回 NULL,并設置 errno 以指示錯誤原因。


以下是一個簡單的示例,演示了如何使用 fopen() 打開文件:


#include
int main() {    FILE *file_ptr;
   // 以只讀模式打開文件    file_ptr = fopen("example.txt", "r");    if (file_ptr == NULL) {        printf("Failed to open file.n");        return 1;    }
   // 在這里可以進行文件讀取操作
   // 關閉文件    fclose(file_ptr);
   return 0;}



在這個示例中,我們嘗試以只讀模式打開名為 example.txt 的文件。如果文件打開失敗,則會打印一條消息并退出程序。否則,我們可以在之后的代碼中對文件進行讀取操作。最后,我們使用 fclose() 函數關閉文件,釋放資源。



2.3、fwrite寫文件


fwrite() 是C語言標準庫中用于向文件寫入數據的函數之一。它的原型如下:


size_t fwrite(const void *ptr, size_t size, size_t nmemb, FILE *stream);


函數參數和返回值含義如下:


  • ptr:參數 ptr 是一個指向緩沖區的指針,該緩沖區中存儲了要寫入到文件中的數據。函數將會把這個緩沖區中的數據寫入到文件中。

  • size:參數 size 指定了每個數據項的字節大小,即每次寫入的數據的大小。

  • nmemb:參數 nmemb 指定了寫入的數據項的個數,即要寫入到文件中的數據項的數量。

  • stream:參數 stream 是一個指向 FILE 結構的指針,它標識了要寫入數據的文件。

  • 返回值:調用成功時,fwrite() 函數返回實際成功寫入到文件中的數據項的數目。如果發生錯誤,則返回值可能小于參數 nmemb(或者等于 0)。

fwrite() 函數返回成功寫入的數據項數目,如果返回值與 nmemb 不同,則表示寫入出現了錯誤。


以下是一個簡單的示例,演示了如何使用 fwrite() 向文件寫入數據:


#include
int main() {    FILE *file_ptr;    char buffer[] = "Hello, world!";
   // 打開文件以便寫入    file_ptr = fopen("example.txt", "w");    if (file_ptr == NULL) {        printf("Failed to open file.n");        return 1;    }
   // 向文件寫入數據    size_t num_written = fwrite(buffer, sizeof(char), sizeof(buffer), file_ptr);    if (num_written != sizeof(buffer)) {        printf("Failed to write to file.n");        fclose(file_ptr);        return 1;    }
   // 關閉文件    fclose(file_ptr);
   return 0;}


在這個示例中,我們向文件 "example.txt" 寫入了字符串 "Hello, world!"。首先我們打開文件以便寫入,然后使用 fwrite() 函數將數據寫入文件,最后關閉文件。


2.4、fread讀文件

fread() 是C語言標準庫中用于從文件讀取數據的函數之一。它的原型如下:


size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream);


函數參數和返回值含義如下:


  • ptr:fread() 函數將讀取到的數據存放在參數 ptr 指向的緩沖區中。這個緩沖區是用來存儲從文件中讀取的數據的。

  • size:fread() 函數從文件讀取 nmemb 個數據項,每個數據項的大小為 size 個字節。因此,總共讀取的數據大小為 nmemb * size 個字節。

  • nmemb:參數 nmemb 指定了要讀取的數據項的個數。

  • stream:參數 stream 是一個指向 FILE 結構的指針,它標識了要從中讀取數據的文件。

  • 返回值:調用成功時,fread() 函數返回成功讀取到的數據項的數目。如果發生錯誤或到達文件末尾,則返回值可能小于參數 nmemb。由于 fread() 無法區分文件結尾和錯誤,返回值小于 nmemb 時,可以使用 ferror() 或 feof() 函數來進一步判斷是發生了錯誤還是已經到達了文件末尾。



fread() 函數返回成功讀取的數據項數目,如果返回值與 nmemb 不同,則表示讀取出現了錯誤。


以下是一個簡單的示例,演示了如何使用 fread() 從文件中讀取數據:


#include
int main() {    FILE *file_ptr;    char buffer[100]; // 緩沖區用于存儲讀取的數據
   // 打開文件以便讀取    file_ptr = fopen("example.txt", "r");    if (file_ptr == NULL) {        printf("Failed to open file.n");        return 1;    }
   // 從文件讀取數據    size_t num_read = fread(buffer, sizeof(char), sizeof(buffer), file_ptr);    if (num_read == 0) {        printf("Failed to read from file.n");        fclose(file_ptr);        return 1;    }
   // 輸出讀取的數據    printf("Read from file: %sn", buffer);
   // 關閉文件    fclose(file_ptr);
   return 0;}


在這個示例中,我們打開了一個名為 "example.txt" 的文件以便讀取。我們使用 fread() 函數從文件中讀取數據,并將其存儲在名為 buffer 的緩沖區中。最后,我們打印出讀取到的數據,并關閉文件。


2.5、fclose關閉文件


fclose() 是C語言標準庫中用于關閉文件的函數之一。它的原型如下:


int fclose(FILE *stream);


函數參數和返回值含義如下:


  • stream:指向 FILE 結構的指針,標識要關閉的文件。

  • 返回值:調用成功返回 0;失敗將返回 EOF(也就是-1)。


以下是一個示例,演示了如何使用 fclose() 關閉文件:


#include
int main() {    FILE *file_ptr;
   // 打開文件以便讀取    file_ptr = fopen("example.txt", "r");    if (file_ptr == NULL) {        printf("Failed to open file.n");        return 1;    }
   // 在這里可以進行文件讀取操作...
   // 關閉文件    if (fclose(file_ptr) != 0) {        printf("Failed to close file.n");        return 1;    }
   return 0;}


在這個示例中,我們打開了一個名為 "example.txt" 的文件以便讀取。在文件讀取操作完成后,我們使用 fclose() 函數關閉了文件。如果關閉文件失敗,則會打印一條錯誤消息并退出程序。


*博客內容為網友個人發布,僅代表博主個人觀點,如有侵權請聯系工作人員刪除。



關鍵詞: 嵌入式 Linux

相關推薦

技術專區

關閉