C语言:lseek函数-----改变文件偏移量

1.文件偏移

  • 通常调用read或write每读写一个文件,就会改变文件的读写位置。在linux中同样可以使用lseek函数来修改文件偏移量,即读写位置。
  • 标准C库的fseek函数和系统函数lseek比较类似,fseek函数也可以移动当前读写位置(或者叫偏移量),其实fseek就是对lseek系统函数封装后实现的,快速回忆一下之前学习的fseek函数的作用及常用参数。
lseek函数
  • 函数原型:
#include <sys/types.h>
#include <unistd.h>
off_t  lseek(int fd, off_t offset, int whence);
  • 参数说明:

  • fd : 文件描述符

  • offset : 文件偏移量(offset为负值表示往前偏移,正值则表示往后偏移)

  • whence : 偏移位置(SEEK_SET,SEEK_CUR,SEEK_END)
      1. SEEK_SET表示从文件开头位置往后移动offset个字节,且offset只能为正数,如果offset为负数是没有意义的。
      2. SEEK_CUR表示从当前文件的位置往前或往后移动offset个字节,如果offset为正数则表示往后移动,为负数则表示往前移动。
      3. SEEK=END表示从文件末尾的位置往前或往后移动offset个字节,如果offset为正数则表示往后移动为负数则表示往前移动。
      在这里插入图片描述


这里给一些lseek函数的一些例子,注释说明了文件偏移量到的具体位置:
  lseek(fd , 0 , SEEK_SET); //文件开始位置
  lseek(fd , 0 , SEEK_END); //文件末尾位置
  lseek(fd , 10 , SEEK_END); //从文件末尾往后移动10个字节
  lseek(fd , -10 , SEEK_END); //从文件末尾往前移动10个字节
  lseek(fd , 100 , SEEK_SET); //从文件开始往后移动100个字节

返回值说明:
  成功返回文件当前读写位置相对于文件开始位置的偏移量(字节数),如果文件读写的位置是在末尾的话,返回值就是文件头与文件尾之间的字节数,也就是文件大小。失败返回-1并设置errno
  
注意几点:
  1. 如果文件偏移量往回超出文件头位置,则返回-1,文件指针不变,还是处于原来的位置。
  2. lseek并不适用与所有的文件类型,也就是说在管道,FIFO,socket或终端不能使用lseek函数,一旦调用将会失败,并设置errno为EPIPE。

  • lseek示例程序:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <fcntl.h>

int main(void) 
{
	int fd, n;
	char buf[] = "hello world test \n";
	char ch;

	fd = open("test.txt", O_RDWR|O_CREAT, 0644);
	if(fd < 0)
	{
		perror("open lseek.txt error");
		exit(1);
	}
	//使用fd对打开的文件进行写操作,写完后文件指针位置位于文件结尾处
	write(fd, buf, strlen(buf));   
	/*
    注意:读和写操作使用同一偏移位置,由于文件指针位于末尾,因此后面的read就会读不到数据了
	这一行的目的是把文件指针移动到文件头位置,这样下面的代码就能读到数据了
	*/
	lseek(fd , 0 , SEEK_SET);
	if(n == -1)
	{
	    perror("lseek fail");
	    exit(1);
	}
	//lseek移到文件开头,read就能读取到数据了
	while((n = read(fd, &ch, 1)))
	{
		if(n < 0)
		{
			perror("read error");
			exit(1);
		}
		//将文件内容按字节读出,写到屏幕
		write(STDOUT_FILENO, &ch, n);  
	}
	close(fd);
	return 0;
}
  • 通过lseek计算文件大小
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <fcntl.h>

int main(void)
{
	int fd;
	char buf[] = "hello world"; 
	fd = open("lseek.txt", O_RDWR | O_CREAT, 0664);
	if(fd < 0){
		perror("open lseek.txt error");
		exit(1);
	}
	write(fd , buf , strlen(buf));
	//lseek函数会从起始位置开始算到文件末尾,就能求出文件的长度了
	int len = lseek(fd, 0, SEEK_END);
	if(len == -1){
		perror("lseek error");
		exit(1);
	}
	//lseek函数的返回值就是文件的大小
	printf("file len = %d\n", len);
	close(fd);
	return 0;
}

  • 扩展文件大小

    lseek函数还可用于扩展文件大小,但是单独使用lseek是不能达到扩展文件的,还必须发生一次IO操作才能扩展文件大小。

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

int main(void)
{
        int fd, n;
        char msg[] = "It's a test for lseek\n";
        char ch;

        fd = open("lseek.txt", O_RDWR|O_CREAT|O_TRUNC, 0644);
        if(fd < 0){
                perror("open lseek.txt error");
                exit(1);
        }

        //扩展文件大小100字节
        int ret = lseek(fd, 100, SEEK_SET);
        if (ret == -1) {
                perror("lseek error");
                exit(1);
        }
        //发生一次I/O操作
        write(fd, "\0", 1);  
        //位移到文件尾部
        int len = lseek(fd, 0, SEEK_END);
        printf("file size is %d\n", len);
        close(fd);
        return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值