linuxopen文件偏移

linuxopen文件偏移,第1张

通过函数lseek可以改变文件当前的读写位置

函数原型

#include <sys/types.h>

#include <unistd.h>

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

参数属性:

offset:表示从文件的whence位置开始偏移的位置大小。

whence:表示文件偏移的位置

有三个选项:

SEEK_SET:表示从文件开始位置偏

SEEK_CUR:表示从文件当前的读写位置偏

SEEK_END:表示从文件的结束位置偏移seek.。

int main

//打开文件

int fd = open("b.bat",O_CREAT|O_RDWR)

if(fd == -1)

perror("错误:"),exit(-1)

//准备数据

char *str = "abcdefghigklmn"

//写入数据

write(fd,str,strlen(str))

//准备空间

char buf[32] = {0}

//将文件的读写位置移动到文件的开始

lseek(fd,0,SEEK_SET)

//读取5个字节的数据

read(fd,buf,5)

//输出

puts(buf)

//关闭文件

close(fd)

文件偏移量的移动时机:

1、lseek主动移动偏移量

2、read会移动偏移量

3、write会移动偏移量

进程 --》运行--》进程在内存中存在一个进程表项(文件表) //文件表在内核管理的内存空间中

文件最开始运行时,有三个“文件描述符”已经打开:0,1,2

当使用open函数打开一个文件的时候,系统就会创建一个文件表

文件表的内容:

1、文件状态标志:O_RDONLY \O_WRONLY...

2、文件偏移量(刚打开的文件偏移量为0)

3、V节点指针:指向“v节点表” //V节点表(磁盘的文件系统上):V节点信息、I节点信息、文件大小

下面是linux系统约定不同类型文件默认的颜色:

白色:表示普通文件

蓝色:表示目录

绿色:表示可执行文件

红色:表示压缩文件

浅蓝色:链接文件

红色闪烁:表示链接的文件有问题

黄色:表示设备文件

灰色:表示其它文件

在linux中,我们需要复制文件描述符,下面是我对文件描述符的理解

int dup(int fd)// 复制一个已经存在的文件描述符,如果成功,返回复制成功后的文件描述符,失败返回-1

int dup(int fd, int fd2)// 复制一个文件描述符,fd表示已经存在的打开的文件描述符,fd2是指定新的文件描述符,如果fd2等于fd,则直接返回,如果fd2存在并且打开,则先close(fd2)后,重新打开,这样fd2和fd就指向了同一个文件(共享打开的文件),如果fd2不存在或者没有打开,则打开fd2,并且指向fd所指向的文件。函数的返回值和fd2一致。该函数可以实现文件的重定位。

每个进程都有一个文件描述符表,每个描述符占用一个描述符项,每个文件描述符可以描述成这样

struct fileDescription {

int index

void *pointer

}

除了整形,还有一个指针,指向文件表,内核为所有打开文件维持一张文件表,每个文件表项包含:

1)文件状态标志(读、写、添写、同步和非阻塞等)

2)当前文件的偏移量

3)指向该文件v节点表项的指针

v节点表中包含了文件所有者、文件长度、文件所在的设备、指向文件实际数据块在磁盘上所在位置的指针等等

下面给出一个例子:

#include <iostream>

#include <stdio.h>

#include <fcntl.h>

#include <sys/stat.h>

#include <unistd.h>

#include <string.h>

using namespace std

int main()

{

cout <<"Hello world!" <<endl

int fd = open("a.txt", O_RDWR | O_CREAT | O_APPEND, S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH)

if (fd <0)

{

printf("open a.txt failed, fd = %d\n", fd)

return -1

}

printf("open a.txt success, fd = %d\n", fd)

fflush(stdout)

// 复制标准输出的文件描述符

int nfd = dup(STDOUT_FILENO)

int fileFd = dup2(fd, STDOUT_FILENO)

if (fileFd <0)

{

printf("dup2 stdout_fileno failed, fileFd = %d\n", fileFd)

return -1

}

printf("重定向标准输出到a.txt, fileFd = %d\n", fileFd)

// 将STDOUT_FILENO复制到fd后,fd并没有发生变化,依然可以通过它写入到a.txt中

const char *pStr = "write string through fd\n"

write(fd, pStr, strlen(pStr))

fileFd = dup2(nfd, fileFd)

if (fileFd <0)

{

printf("dup2 reback stdout_fileno failed\n")

return -1

}

printf("print back to stdout standard, fileFd = %d\n", fileFd)

// 上面使用dup2的时候,STDOUT_FILENO是存在并打开的,我们来测试下不存在的文件描述符的情况

fileFd = dup2(fd, 20)

if (fileFd <0)

{

printf("dup2不存在的文件描述符失败, fileFd = %d\n", fileFd)

}

else

{

printf("dup2不存在的文件描述符成功, fileFd = %d\n", fileFd)

const char *pStr = "write string through fileFd\n"

write(fileFd, pStr, strlen(pStr))

}

close(nfd)

close(fileFd)

return 0

}

文件指针是关键,标志两个文件描述符是否一致,看文件指针是否一致即可 ,如果两个或者多个文件描述符指向同一个文件表,那么对他们的 *** 作是对同一个文件进行 *** 作


欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/yw/8349761.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-04-15
下一篇2023-04-15

发表评论

登录后才能评论

评论列表(0条)

    保存