函数原型:
#include<unistd.h>
ssize_t read(int fd, void *buf, size_t count);返回值:读到的字节数,若已到文件尾,返回0;若出错,返回-1
参数:
- fd:函数open的返回值
- buf:缓冲区,存储要读取的数据
- count:缓冲区的最大字节数size(buf)
函数原型:
#include<unistd.h>
sszie_t write(int fd, const void *buf, size_t count);返回值:若成功,返回已写的字节数,若出错,返回-1
参数:
- fd:函数open返回值
- buf:要写到文件的数据
- count:strlen(buf)
测试代码:
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>int main(int argc, const char* argv[])
{int fd = open("english.txt", O_RDWR);printf(" fd = %d\n", fd);int fd1 = open("temp", O_WRONLY | O_CREAT, 0664);printf("fd1 = %d\n",fd1);char buf[4096];int len = read(fd, buf, sizeof(buf));while(len > 0) {int ret = write(fd1, buf, len);printf("ret = %d\n", ret);len = read(fd, buf, sizeof(buf));}close(fd);close(fd1);return 0;
}
输出结果:
函数原型:
#include<unistd.h>
off_t lssk(int fd, off_t offset, int whence);返回值:若成功, 返回新的文件的偏移量;若出错,返回-1
参数:
- 若whence是SEEK_SET,则将该文件的偏移量设置为距文件的开始处offset个字节。
- 若whence是SEEK_CUR,则将该文件的偏移量设置为当前值加offset,offset可正可负。
- 若whence是SEEK_END,则将该文件的偏移量设置为文件长度加offset,offset可正可负。
使用:
- 文件指针移动到头部:lseek(fd, 0, SEEK_SET);
- 获取文件指针当前位置:int len = lseek(fd, 0, SEEK_CUR);
- 获取文件长度:int len = lseek(fd, 0, SEEK_END);
- 文件拓展:文件原大小100k,拓展为1100k。lseek(fd, 1000, SEEK_END); 最后一次写操作:write(fd, "a", 1);
测试代码:
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <string.h>
#include <fcntl.h>int main(int argc, const char *argv[])
{int fd = open("english.txt", O_RDWR);if(fd == -1) {perror("open error");exit(1);}int len = lseek(fd, 1000, SEEK_END);printf("len = %d\n", len);write(fd, "a", 1);close(fd);return 0;
}
输出结果:
阻塞与非阻塞
阻塞和非阻塞是文件的属性还是read函数的属性? 答案:文件的属性
- 普通文件:hello.c 不阻塞
- 终端设备:/dev/tty、管道、套接字:,默认阻塞
1. 阻塞读终端
测试代码:
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>int main()
{char buf[10];int n;n = read(STDIN_FILENO, buf, 10);if(n < 0) {perror("read STDIN_FILENO");exit(1);}write(STDOUT_FILENO, buf, n);return 0;
}
输出结果:
分析:
测试代码:
#include <unistd.h>
#include <fcntl.h>
#include <string.h>
#include <stdio.h>
#include <errno.h>
#include <stdlib.h>
#define MSG_TRY "try agin\n"int main()
{char buf[10];int fd, n;// /dev/tty --->当前打开的终端设备fd = open("/dev/tty", O_RDONLY | O_NONBLOCK);if(n < 0) {perror("open /dev/tty");exit(1);}tryagin:n = read(fd, buf, 10);if(n < 0) {//如果write为非阻塞,但是没有数据可读,此时全局变量errno被设置为EAGAINif(errno == EAGAIN) {sleep(3);write(STDOUT_FILENO, MSG_TRY, strlen(MSG_TRY));goto tryagin;}perror("read /dev/tty");exit(1);}write(STDOUT_FILENO, buf, n);close(fd);return 0;
}
输出结果: