# 打开、关闭和读写文件
一、打开和关闭文件
int open(char *filename,int flags,mode_t mode),打开文件
open函数将filename转换为一个文件描述符
flags参数也可以是一个或者更多位掩盖的
mode参数指定了新文件的访问权限位
int close(int fd),关闭文件
例子:
int main()
{
int fd1,fd2;
fd1=open("foo.txt",O_RDONLY,0);
close(fd1);
fd2=open("baz.txt",O_RDOLY,0);
printf("fd2=%d\n",fd2);
exit(0);
}
输出:
fd=3
解析:因为在进程中描述符已经占了0、1、2,而opern函数返回的描述符总是在进程中当前没有打开的最小描述符。第一次打开文件时,fd1=3,但调用了close 函数关闭文件,所以并没有影响后面的fd2=3;
二:读和写文件
ssize_t read(int fd,void *buf,size_t n),读文件
ssize_t write(int fd,const void *buf,size_t n),写文件
其中size_t被定义为un_signed long,而ssize_t被定义为long。
三、综合
例1
int main()
{
int fd1,fd2;
char c;
fd1=open("foobar.text",O_RDONLY,0);
fd2=open("foobar.text",O_RDOLY,0);
read(fd1,&c,1);
read(fd2,&c,2);
printf("c=%c\n",c);
exit(0);
}
输出:
c=f
解析:
打开一个文件,光标从文件第一个开始,所以即使打开了两次文件,读到的都只是第一个字母所以c=f
例2
int main()
{
int fd;
char c;
fd=open("foobar.text",O_RDOLY,0);
iF(fork()==0){
read(fd,&c,1);
exit(0);
}
wait(NULL);
read(fd,&c,1);
printf("c=%c\n",c);
exit(0);
}
输出:
c=o
解析:
根据fork()函数的特点,子进程与父进程公用资源,则打开一次文件,父进程读取第一个字母,子进程就会读第二个字母,所以输出c=o
四、I/O重定向
int dup2(int oldfd,int newfd)
dup2函数复制描述符表表项oldfd到描述符表表项newfd,覆盖描述符表表项newfd以前的内容
例1
int main(0
{
int fd1,fd2;
char c;
fd1=open("foobar.tet",O_RDONLY,0)
fd2=open("foobar.tex",O_RDONLY,0);
read(fd2,&c,1);
dup2(fd2,fd1);
read(fd1,&c,1);
printf(c=%c\n",c);
exit(0);
}
输出.;
c=o
解析:打开了两次文件,应该是都指向文件的第一个字母,但是由于dup2的覆盖效果,fd2覆盖了fd1的内容,则读取fd1的内容实则是读fd2的内容,所以光标下移,输出第二个字母
例2
int main(int argeint argc, char *argv[])
{
int fd1, fd2, fd3;
char c1, c2, c3;
char *fname = argv[1];
fd1 = Open(fname, O_RDONLY, 0);
fd2 = Open(fname, O_RDONLY, 0);
fd3 = Open(fname, O_RDONLY, 0);
dup2(fd2, fd3);
Read(fd1, &c1, 1);
Read(fd2, &c2, 1);
Read(fd3, &c3, 1);
printf("c1 = %c, c2 = %c, c3 = %c\n", c1, c2, c3);
Close(fd1);
Close(fd2);
Close(fd3);
return 0;
}
文件为:“abcdef”
输出:
c1=a,c2=a,c3=b
解析:打开三次文件,光标都在文件的第一个字母,但由于fd2的内容覆盖了fd3的内容,所以在读fd3的内容时读出的应该是fd2的内容,则输出的c1是文件中的第一个字母,c2也是文件中的第一个字母,c3就是c2后的字母,即文件的第二个字母