write之后为啥read不出来 为啥?
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了write之后为啥read不出来 为啥?相关的知识,希望对你有一定的参考价值。
#include <stdio.h>
#include <fcntl.h>
#include <string.h>
int main ()
int fd = open("tmp", O_CREAT | O_RDWR | O_TRUNC, 0664);
if(fd < 0)
perror("open");
goto err_perror;
char buf[] = "hello";
write(fd, buf, sizeof(buf));
memset(buf, 0, sizeof(buf));
read(fd, buf, sizeof(buf));
printf("%s\n", buf);
return 0;
err_perror:
return -1;
不write直接read可以读出来,write之后read就读不出来了。
为什么。
你在memset()后面加上lseek(fd, 0, 0);
就能读出来了;
功 能: 移动文件读/写指针
头文件:#include <sys/types.h> #include <unistd.h>
用 法: off_t lseek(int handle, off_t offset, int fromwhere);
所有打开的文件都有一个当前文件偏移量(current file offset),以下简称为 cfo。cfo 通常是一个非负整数,用于表明文件开始处到文件当前位置的字节数。读写操作通常开始于 cfo,并且使 cfo 增大,增量为读写的字节数。文件被打开时,cfo 会被初始化为 0,除非使用了 O_APPEND 。
使用 lseek 函数可以改变文件的 cfo 。
lseek 的以下用法返回当前的偏移量:
off_t currpos;
currpos = lseek(fd, 0, SEEK_CUR);
这个技巧也可用于判断我们是否可以改变某个文件的偏移量。如果参数 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)决定的。
参数 whence为下列其中一种:(SEEK_SET,SEEK_CUR和SEEK_END和依次为0,1和2).
SEEK_SET 将读写位置指向文件头后再增加offset个位移量。
SEEK_CUR 以目前的读写位置往后增加offset个位移量。
SEEK_END 将读写位置指向文件尾后再增加offset个位移量。
当whence 值为SEEK_CUR 或SEEK_END时,参数offet允许负值的出现。
下列是教特别的使用方式:
1) 欲将读写位置移到文件开头时:
lseek(int fildes,0,SEEK_SET);
2) 欲将读写位置移到文件尾时:
lseek(int fildes,0,SEEK_END);
3) 想要取得目前文件位置时:
lseek(int fildes,0,SEEK_CUR);返回值 当调用成功时则返回目前的读写位置,也就是距离文件开头多少个字节。若有错误则返回-1,errno 会存放错误代码。 参考技术A write(fd, buf, sizeof(buf));
memset(buf, 0, sizeof(buf));
在这你得把文件指针通过seek函数往回倒sizeof(buf)个字节,
read(fd, buf, sizeof(buf));追问
我试过用lseek 置到SEEK_SET 然后再读
我再试看sizeof(buf)看看。
以上是关于write之后为啥read不出来 为啥?的主要内容,如果未能解决你的问题,请参考以下文章
为啥 BufferedStream.Write 会抛出“此流不支持查找操作”?
document.write 引用script标签 为啥不起作用???