Linux 通過lseek()來實現檔案大小的設定

2puT發表於2016-07-09

一 函式介紹:

函式名: lseek()

功 能: 移動檔案讀/寫指標

所需標頭檔案:

#include <sys/types.h>

#include <unistd.h>

函式原型:

off_t lseek(int fd, off_t offset, int whence);

重新定位已開啟的檔案的偏移量,與whence的取值有關;

引數:

fd:檔案描述符,對應已經開啟的檔案;

offset:指出偏移量;

whence:指出偏移的方式,取值如下:

SEEK_SET:偏移到offset位置處(相對檔案頭)

SEEK_CUR:偏移到當前位置+offset位置處;

SEEK_END:偏移到檔案尾+offset位置處;

返回值:

呼叫成功則返回最終的偏移量(從檔案頭開始數);

呼叫失敗則返回-1,並設定相應的errno;

二 巧妙利用

1. 返回當前的偏移量

off_t currpos;

currpos = lseek(fd, 0, SEEK_CUR);

2. 返回檔案大小

off_t currpos;

currpos = lseek(fd, 0, SEEK_END);

3. 擴充檔案大小

lseek()方法允許偏移

  這個技巧也可用於判斷我們是否可以改變某個檔案的偏移量。如果引數 fd(檔案描述符)指定的是 pipe(管道)、FIFO 或者 socket,lseek 返回 -1 並且置 errno 為 ESPIPE。
  對於普通檔案(regular file),cfo 是一個非負整數。但對於特殊裝置,cfo 有可能是負數。因此,我們不能簡單地測試 lseek 的返回值是否小於 0 來判斷 lseek 成功與否,而應該測試 lseek 的返回值是否等於 -1 來判斷 lseek 成功與否。
  lseek 僅將 cfo 儲存於核心中,不會導致任何 I/O 操作。這個 cfo 將被用於之後的讀寫操作。

  如果 offset 比檔案的當前長度更大,下一個寫操作就會把檔案“撐大(extend)”。這就是所謂的在檔案裡創造“空洞(hole)”。沒有被實際寫入檔案的所有位元組由重複的 0 表示。空洞是否佔用硬碟空間是由檔案系統(file system)決定的。

概念補充:

當前檔案偏移量(current file offset),以下簡稱為 cfo。cfo 通常是一個非負整數,用於表明檔案開始處到檔案當前位置的位元組數。讀寫操作通常開始於 cfo,並且使 cfo 增大,增量為讀寫的位元組數。檔案被開啟時,cfo 會被初始化為 0,除非使用了 O_APPEND 。

相關文章