国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統 > Unix > 正文

UNIX高級環境編程(1)File I/O

2024-06-28 13:21:59
字體:
來源:轉載
供稿:網友
UNIX高級環境編程(1)File I/O引言:

Unix系統中主要的文件操作包括:

  • open
  • read
  • write
  • lseek
  • close

unbuffered IO和standard I/O相對應,后面的章節我們會討論這兩者的區別。

在討論open函數的時候,會引入原子操作,多進程通信(共享文件描述符)和內核相關的數據結構。

一,文件描述符

對應內核來說,每一個打開的文件都對應一個非負整數。

有三個特殊的文件描述符:

  • 0表示標準輸入
  • 1表示標準輸出
  • 2表示標準錯誤輸出

對于較新的內核來說(linux3.2.0,Solaris10等),文件描述符的數量并沒有明確的限制,受限于內存的大小。

二,常用的幾個文件操作函數

常用的文件操作函數包括:open,read,write,lseek,close

1 open和openat函數

函數聲明:

#include <fcntl.h>int open (const char *path, int oflag, … /* mode_t mode */);int openat (int fd, const char *path, int oflag, … /* mode_t mode */);

返回值:

OK:文件描述符(非負整數)

Error:-1

注:參數列表中,“...”表示不同的系統和標準中,該處的參數可能不相同。

參數說明:

path:文件名

oflag:打開創建文件的屬性。

下面有五個必選的oflag參數值,這五個值有切只能選一個。另外還有若干個可選參數值,可以自行百度。

NewImage

細節說明:

由open和openat返回的文件描述符保證為未使用的最小的文件描述。有的應用利用這一特性,先關閉標準輸入描述符0,就可以在標準輸入描述0上打開文件。

參數fd可以區分open和openat函數。其取值有三種可能:

  1. path表示一個絕對路徑,則fd參數無用,openat的功能和open相同;
  2. path表示一個相對路徑,則fd是一個文件描述符,指定了path在文件系統中的起始位置,fd為打開path父目錄時獲取的文件描述符;
  3. path表示一個相對路徑,而fd的值為AT_FDCWD,這時,path的父目錄為當前工作目錄,openat和open的功能相同。

openat函數解決了兩個問題:

  • 在多線程條件下,默認各個線程的工作目錄時相同的(當前工作目錄),使用這個函數可以使得各個線程的指定不同的工作目錄;
  • 提供了一種方法解決TOCTTOU(time-of-check-to-time-of-use) error。

這里介紹一下TOCTTOU錯誤。該類錯誤指的是,程序是非常脆弱的(vulnerable)如果該程序調用了兩個文件相關的函數,第二個函數依賴于第一個函數的結果。因為兩個函數是非原子操作,被操作的文件可能被兩個函數輪流操作(線程切換),導致第一個函數的結果出錯,從而程序出錯。

2 creat函數

函數聲明:

#include <fcntl.h>int creat(const char* path, mode_t mode);

返回值:

  • OK:文件描述符(只寫)
  • Error:-1

creat函數相當于下面這樣調用open函數

open (path, O_WEONLY | O_CREAT | O_TRUNC, mode);

creat有一點不方便,因為它打開的文件描述符是只讀的,如果希望寫入之后讀回,需要依次調用creat、close和open,才能實現。

因此,在這種場景下,一個更好的打開文件的方法是像下面這樣調用open函數:

open (path, O_RDWR | O_CREAT | O_TRUNC, mode);

3 close函數

函數聲明:

#include <unistd.h>int close(int fd);

返回值:

  • 0 :OK
  • -1:Error

關閉一個文件會釋放所有當前進程加在該文件上的記錄鎖。

4 lseek函數

每一個打開的文件都有一個”當前文件偏移量(current file offset)“,該偏移量是一個非負整數,記錄了從文件開始到當前位置的字節數。

函數聲明:

#include <unistd.h>off_t lseek(int fd, off_t offset, int whence);

參數說明:

offset的作用取決于參數whence的值:

  • 如果whence的值是SEEK_SET,文件的偏移量設置為offset個字節;
  • 如果whence的值是SEEK_CUR,文件的偏移量設置為當前偏移量加上參數offset的值;
  • 如果whence的值是SEEK_END,文件的偏移量設置為文件長度加上參數offset的值,offset可以是正值或負值。

細節說明:

獲取當前文件偏移量的方法:

1 off_t currpos;2 3 currpos = lseek(fd, 0, SEEK_CUR);

lseek只記錄當前文件在內核中的偏移量,并不會引起任何的IO操作。返回的offset將會在后面的read或write函數中使用。

偏移量可以比當前文件的長度大,這時,再調用write函數時,將擴展該文件的長度。這樣的操作相當于在文件中建了一個洞,該洞范圍內讀時返回0。

使用od命令可以看到文件中的hole

NewImage

4 read函數

函數聲明:

#include <unistd.h>ssize_t read(int fd, void *buf, size_t nbytes);

返回值:

  • 正整數:讀入的字節數
  • 0:文件結尾
  • -1: error

細節說明:

在一些情況下,函數返回的字節數比指定的讀入字節數要小,多數是因為讀到了文件末尾,或者指定的讀取位置中包含的字節數小于指定的讀入字節數,這時,read返回的為可讀到的字節數。

5 write函數

函數聲明:

#include <unistd.h>ssize_t write (int fd, const void *buf, size_t nbytes);

返回值:

  • 非負整數:寫入的字節數,OK
  • -1: Error

返回值總是等于參數nbytes的值,否則就會報錯。

對于常規的文件來說,寫操作總是從當前文件偏移量開始。

三、小結

簡單地介紹了一下常用的文件IO操作,并介紹了一些使用上的細節,比較常規。

下一篇講介紹更多文件IO的特性,包括:dup,fcntl,sync,fsync和ioctl函數。。

好久沒寫博客了,又第一次用mac下的一個博客軟件寫,不太熟悉,所以寫的比較簡單,以后會寫的更努力。

參考資料:

《Advanced PRogramming in the UNIX Envinronment 3rd》


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 汝阳县| 蒲城县| 吉木乃县| 广灵县| 襄汾县| 克山县| 格尔木市| 新绛县| 察雅县| 禹城市| 淮北市| 光山县| 柘城县| 武清区| 阿图什市| 黑水县| 板桥市| 屯门区| 盐津县| 康马县| 蓬安县| 扶风县| 沙湾县| 内江市| 沾化县| 繁峙县| 合山市| 景谷| 恩施市| 南京市| 贵州省| 根河市| 海安县| 新邵县| 天峻县| 怀集县| 沛县| 昭觉县| 黑龙江省| 南投县| 常山县|