3.1 前言
unix环境下的I/O函数主要功能为打开文件、读文件、写文件等,主要包括open、close、read、write、lseek。本章的函数被称之为不带缓存的I/O——不带缓存是之每个read和write都调用一个系统调用。本章将对文件I/O相关的幻术,多进程之间共享文件等问题进行讨论。
3.2 文件描述符
文件描述符是一个非负的整数,所有打开的文件由内核来引用。UNIX shell使得文件描述符0与进程标准输入关联,文件描述符1与系统标准输出关联,文件描述符2与标准出错向关联。在posix应用程序中,幻数0、1、2 被替代为STDIN_FILENO、STDOUT_FILENO、STDERR_FILENO,这些常熟被定义在<unistd.h>中
3.3 文件操作函数
3.3.1 open函数
open函数可以打开或者创建一个文件
int open(const char *pathname,int oflag,.../*, mode_t mode* /)
返回:成功返回文件描述符,失败返回-1
对于open函数仅当创建新文件时才使用第三个参数。
oflag参数用来说明函数的多个选择项,通常定义在<fcntl.h>
- O_RDONLY:只读打开
- O_WRONLY:只写打开
- O_RDWR:读写打开
上面参数通常指定一个,其他参数可以附加,参数之间采用或运算
- O_APPEND:每次写文件加到文件末尾
- O_CREAT:若文件不存在则创建它,使用此选线需要同时说明第三个参数来表示该文件的存取许可权位
- O_EXCL:如果同时指定个O_CREAT,文件却已经存在,则出错,用来测试文件是否存在
- O_TRUNC:如果文件存在,并且为只读或者只写打开,则将其长度截断为0
- O_NOCTTY:如果pathname是终端设备,则不将设备分配作为此进程的控制终端
- O_NONBLOCK:如果pathname 指定是一个FIFO,一个块文件,或者一个字符特殊文件,则本次打开操作和后续的I/O操作设置为非阻塞方式
- O_SYNC 使每次write都等到物理I/O完成
3.3.2 creat函数
int creat(const char *pathname,mode_t mode
返回:成功返回文件描述符,失败返回-1
此函数等同与open(pathname,O_WRONLY|O_CREAT|O_TRUNC,mode);
mode为文件存取许可权,后续将说明如何指定mode
3.3.3 close函数
int close(int filedes)
返回:成功为0,出错为-1
关闭文件会释放该进程加在该文件上的所有记录锁,此外,当一个进程终止时,内核会关闭属于该进程的文件
3.3.4 lseek函数
off_t lseek(int filedes,off_t offset,int whence);
返回:成功为文件新的位移量,出错返回-1
lseek设置文件当前的位移量。它是一个非负的整数,用于度量从文件开始处计算的字节数。参数offset为相对位移量,whence值是起点,whence值如下:
- SEEK_SET:文件开始处
- SEEK_CUR:当前位置
- SEEK_END:文件末尾
通常文件位移量是一个非负的整数,但是某些设备也允许负的位移量,因此在判断lseek函数的返回值时要谨慎,不要测试其是否<0,而要判断其是否等于-1。
3.3.5 read函数
从打开的文件中读取数据.
ssize_t read(int filedes,void *buff,size_t nbytes);
返回:成功为字节数,若到达文件尾为0,失败为-1
对于read函数要注意有多种情况导致实际读到的字节数少于要求读的字节数
- 读普通文件,读到文件末尾。例如要读100字节,文件还剩30字节,此时返回30,再一次读文件时返回0(文件尾端)
- 从终端设备读时,每次最多读一行
- 从网络读时,网络中的缓冲机构可能导致返回的值小于实际读到的值
- 某些面向记录的设备,如磁带,一次最多返回一个记录
3.3.6 write函数
ssize_t write(int filedes,void *buff,size_t nbytes);
返回:成功为写的实际字节数,出错为-1
write出错的常见原因是:磁盘已经写满,或者超过一个给定进程的文件长度限制
下面代码是对上面函数使用的一个示范,功能是创建一个具有空洞的文件
#include <stdio.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/types.h>
#include <stdlib.h>
#include <errno.h>
#define FILE_MODE (S_IRUSR|S_IWUSR|S_IRGRP|S_IROTH)
char buf1[]="abcdefghij";
char buf2[]="ABCDEFGHIJ";
int main()
{
int fd;
if((fd=open("file.hole",O_RDWR|O_CREAT,FILE_MODE))<0)
{
perror("creat file error:");
exit(-1);
}
if(write(fd,buf1,10)!=10)
{
perror("write buf1 error:");
exit(-1);
}
/* offset now is 10 */
if(lseek(fd,40,SEEK_SET)==-1)
{
perror("lseek error:");
exit(-1);
}
/* offset now is 40 */
if(write(fd,buf2,10)!=10)
{
perror("write buf2 error:");
exit(-1);
}
/* offset now is 50 */
exit(0);
}
下图是程序运行结果,程序创建file.hole的空洞文件,未写的部分被读为0。由图可以看出文件长度为50字节,每一行的第一列七位数是八进制表示的字节位移量,od命令功能是查看文件实际内容