linux系統程式設計之檔案與IO(三):利用lseek()建立空洞檔案

mickole發表於2013-07-10

一、lseek()系統呼叫

功能說明:

通過指定相對於開始位置、當前位置或末尾位置的位元組數來重定位 curp,這取決於 lseek() 函式中指定的位置

函式原型:

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

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

引數說明:

fd:檔案描述符

offset:偏移量,該值可正可負,負值為向前移

whence:搜尋的起始位置,有三個選項:

(1).SEEK_SET: 當前位置為檔案的開頭,新位置為偏移量大小
(2).SEEK_CUR: 當前位置為檔案指標位置,新位置為當前位置加上偏移量大小
(3).SEEK_END: 當前位置為檔案結尾,新位置為偏移量大小

返回值:檔案新的偏移值

二、利用lseek()產生空洞檔案(hole)

說明:

The lseek() function allows the file offset to be set beyond the end of the file (but this does not change the size of the file).  If  data  is later written at this point, subsequent  reads of the data in the gap (a "hole") return null bytes ('\0') until data is  actually  written  into the gap.

程式程式碼:

#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <stdlib.h>

#define ERR_EXIT(m) \
    do \
    { \
        perror(m); \
        exit(EXIT_FAILURE); \
    } while(0)

int main(void)
{
    int fd;
    int ret;
    fd = open("hole.txt",O_WRONLY|O_CREAT|O_TRUNC,0644);
    if(fd == -1)
        ERR_EXIT("open error");
    write(fd,"hello",5);
    ret = lseek(fd,1024*1024*1024,SEEK_CUR);
    if(ret == -1)
        ERR_EXIT("lseek error");
    write(fd,"world",5);
    close(fd);
    return 0;
}

測試結果:

QQ截圖20130710133529

程式建立一個hole檔案,然後寫入”hello”字元,在利用lseek系統呼叫從當前位置偏移到1024*1024*1024之後再寫入”world”字元,ls顯示檔案大小為1.1G,實際上它並沒有佔用這麼多的磁碟空間,du表明hole檔案只有8k,系統只是儲存一些資訊,用它來表示有多少個\0,當我們用cat檢視檔案內容時只看到hello,由於檔案太大最後的world太后以致看不到,但當我們用od命令檢視時,可以發現有好多個\0。

相關文章