一、简述
标准库的I/O函数:
open(), read(), write(), lseek(), close()。
系统I/O和标准I/O区别:
1、移植性
系统函数移植性有限,标准函数移植性高。
2、文件IO层次
系统IO函数为低级,标准IO函数为高级,
区分标准:低级文件IO运行在内核态,高级文件IO运行在用户态。
3、缓冲区
系统函数不带缓冲区,标准函数自带缓冲区。系统函数既可以读写字符、字符串、格式化数据,也可以读写二进制数据。标准函数只能读写二进制文件,但效率高、速度快。
二、系统I/O
open()函数 | |||
功能 | 打开一个指定的文件并获得文件描述符,或者创建一个新文件 | ||
头文件 | #include <sys/types.h> | ||
#include <sys/stat.h> | |||
#include <fcntl.h> | |||
原型 | int open(const char *pathname, int flags) | ||
int open(const char *pathname, int flags, mode_t mode) | |||
参数 | pathname | 要打开的文件路径名称 | |
flags | O_RDONLY | 只读打开 | |
O_WRONLY | 只写打开 | ||
O_RDWR | 可读可写打开 | ||
O_CREAT | 如果文件不存在,就创建该文件 | ||
O_TRUNC | 格式化文件 | ||
O_APPEND | 以追加的方式打开文件 | ||
O_NONBLOCK | 非阻塞 | ||
O_EXCL | 与O_CREAT一起调用,确保调用者创建出文件。防止两个程序同时创建一个文件,如果文件已经存在,open调用将失败并返回-1。 | ||
mode | 新建的文件的权限 | ||
返回值 | 成功 | 返回文件的描述符fd (fd为大于0的整数) | |
失败 | 返回 -1,并且errno会被设置 |
说明:
1、文件描述符
0:标准输入 1:标准输出 2:标准错误
2、flags 参数
flags 的各种取值可以用位或的方式叠加起来
例如 :读写方式打开,不存在要新建,如果存在了则清空。则flags 的取值应该是:O_RDWR | O_CREAT | O_TRUNC。
3、mode(文件的权限)
444 r--r--r-- 600 rw------- 644 rw-r--r-- 666 rw-rw-rw- 700 rwx------
744 rwxr--r-- 755 rwxr-xr-x 777 rwxrwxrwx从左至右,
1-3位数字代表文件所有者的权限,
4-6位数字代表同组用户的权限,
7-9数字代表其他用户的权限。
读取权限:等于4 用 r 表示
写入权限:等于2 用 w 表示
执行权限:等于1 用 x 表示
write()函数 | ||
功能 | 将数据写入指定文件 | |
头文件 | #include <unistd.h> | |
原型 | ssize_t write(int fd, const void *buf, size_t count) | |
参数 | fd | 将写入数据的文件的文件描述符 |
buf | 存储将要写入文件的数据的首地址 | |
count | 要写入的字节数 | |
返回值 | 成功 | 返回实际写入的字节数 |
失败 | 返回 -1,并且errno会被设置 |
read()函数 | ||
功能 | 读取指定文件的数据 | |
头文件 | #include <unistd.h> | |
原型 | ssize_t read(int fd, void *buf, size_t count) | |
参数 | fd | 将读取指定的文件的数据的描述符 |
buf | 存放读取到的数据的内存首地址 | |
count | 要读取的字节数 | |
返回值 | 成功 | 返回实际读取到的字节数 |
失败 | 返回 -1,并且errno会被设置 |
close()函数 | ||
功能 | 关闭指定的文件 | |
头文件 | #include <unistd.h> | |
原型 | int close(int fd) | |
参数 | fd | 将读取指定的文件的数据的描述符 |
返回值 | 成功 | 返回 0 |
失败 | 返回 -1,并且errno会被设置 |
lseek()函数 | ||
功能 | 调整光标位置的偏移量 | |
头文件 | #include <sys/types.h> | |
#include <unistd.h> | ||
原型 | off_t lseek(int fd, off_t offset, int whence) | |
参数 | fd | 要调整光标位置偏移量的文件的描述符 |
offset | 相对基准点的偏移大小 | |
whence (基准点) | SEEK_SET (文件头) | |
SEEK_CUR (当前位置) | ||
SEEK_END (文件尾) | ||
返回值 | 成功 | 当前光标相对文件开头的偏移量 |
失败 | 返回 -1,并且errno会被设置 | |
备注 | 可用 lseek(fd, 0, SEEK_END) 来测量文件大小 |
三、demo
- 创建文件file1 ---------- open()函数
- 往文件中写入数据 ----------- write()函数
- 调整光标偏移量--------- lseek()函数
- 读取文件的数据并输出-------- read()函数
- 关闭文件-------- close()函数
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <string.h>
#include <unistd.h>
int main()
{
int fd;
int n_write;
int n_read;
int len;
char w_buf[128] = "hallo world!";
char r_buf[128] = {0};
int byte;
fd = open("./file1", O_RDWR | O_CREAT | O_TRUNC, 0600);
if (fd == -1) {
perror("error from open");
}
n_write = write(fd, &w_buf, strlen(w_buf));
if (n_write == -1) {
perror("error form wtite");
}
len = lseek(fd, 0,SEEK_CUR);
byte = lseek(fd, 0, SEEK_SET);
if (byte == -1) {
perror("error form lseek");
}
n_read = read(fd, &r_buf, len);
if (n_read == -1) {
perror("error form read");
} else {
printf("%s\n", r_buf);
}
close(fd);
return 0;
}
运行结果: