lseek函数的使用 转载自http://atia.blog.sohu.com/94895919.html

所有打开的文件都有一个当前文件偏移量(current file offset),以下简称为 cfocfo 通常是一个非负整数,用于表明文件开始处到文件当前位置的字节数。读写操作通常开始于 cfo,并且使 cfo 增大,增量为读写的字节数。文件被打开时,cfo 会被初始化为 0,除非使用了 O_APPEND 

    使用 lseek 函数可以改变文件的 cfo 

        #include <unistd.h>

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

 

                    返回值:新的偏移量(成功),-1(失败)

 

参数 offset 的含义取决于参数 whence

    1. 如果 whence 是 SEEK_SET,文件偏移量将被设置为 offset

    2. 如果 whence 是 SEEK_CUR,文件偏移量将被设置为 cfo 加上 offset

       offset 可以为正也可以为负。

    3. 如果 whence 是 SEEK_END,文件偏移量将被设置为文件长度加上 offset

       offset 可以为正也可以为负。

SEEK_SETSEEK_CUR 和 SEEK_END 是 System V 引入的,在这之前使用的是 0和 2

    lseek 的以下用法返回当前的偏移量:

        off_t    currpos;

        currpos = lseek(fd, 0, SEEK_CUR);

这个技巧也可用于判断我们是否可以改变某个文件的偏移量。如果参数 fd(文件描述符)指定的是 pipe(管道)、FIFO 或者 socketlseek 返回 -1 并且置 errno 为 ESPIPE

    对于普通文件(regular file),cfo 是一个非负整数。但对于特殊设备,cfo 有可能是负数。因此,我们不能简单地测试 lseek 的返回值是否小于 来判断 lseek 成功与否,而应该测试 lseek 的返回值是否等于 -1 来判断 lseek 成功与否。

    lseek 仅将 cfo 保存于内核中,不会导致任何 I/O 操作。这个 cfo 将被用于之后的读写操作。

    如果 offset 比文件的当前长度更大,下一个写操作就会把文件“撑大(extend)”。这就是所谓的在文件里创造“空洞(hole)”。没有被实际写入文件的所有字节由重复的 表示。空洞是否占用硬盘空间是由文件系统(file system)决定的。

    以下程序创建一个有空洞的文件:

        /* Standard C header */

        #include <stdio.h>

        /* Unix header */

        #include <fcntl.h>

        #include <unistd.h>

        #include <sys/stat.h>

        char    buf1[] = "abcdefghij";

        char    buf2[] = "ABCDEFGHIJ";

        int main(void)

        {

            int     fd, size;

            if ((fd = creat("file.hole", S_IRUSR|S_IWUSR)) < 0)

            {

                printf("creat error\n");

                return -1;

            }

            size = sizeof buf1 - 1;

            if (write(fd, buf1, size) != size)

            {

                printf("buf1 write error\n");

                return -1;

            }

            /* offset now = 10 */

            if (lseek(fd, 16384, SEEK_SET) == -1)

            {

                printf("lseek error\n");

                return -1;

            }

            /* offset now = 16384 */

            size = sizeof buf2 - 1;

            if (write(fd, buf2, size) != size)

            {

                printf("buf2 write error\n");

                return -1;

            }

            /* offset now = 16394 */

            return 0;

        }

运行结果:

       $ ./a.out

       $ ls -l file.hole            //  查看文件大小

    -rw-r--r-- 1 sar          16394 Nov 25 01:01 file.hole

       $ od -c file.hole            //  查看文件实际内容

    0000000   a  b  c  d  e  f  g  h  i  j \0 \0 \0 \0 \0 \0

    0000020  \0 \0 \0 \0 \0 \0 \0 \0 \0 \0 \0 \0 \0 \0 \0 \0

    *

    0040000   A  B  C  D  E  F  G  H  I  J

    0040012

-c 选项告诉 od 以字符模式显示文件内容。可以看到,没有被实际写入文件的所有字节都由 表示。每行前面的 个数字是文件偏移量的八进制表示形式。

    为了证明 file.hole 的确有空洞,我们拿它和另一个相同大小但没有空洞的文件比较:

           $ ls -ls file.hole file.nohole

          8 -rw-r--r-- 1 sar        16394 Nov 25 01:01 file.hole

         20 -rw-r--r-- 1 sar        16394 Nov 25 01:03 file.nohole

可以看到,没有空洞的 file.nohole 占用了 20 个块(block),而 file.hole 仅占用 个块

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值