Linux中一切都是文件。如普通文件,目录,设备,管道等。
操作这些文件有两种方式,调用系统函数和使用标准I/O库。
一、调用系统函数
1.文件描述符:数值类型,表示打开的文件标识
程序运行时,会首先打开3个文件描述符,0(标准输入文件),1(标准输出文件),2(标准错误文件)
2.系统调用常用函数
1)open函数:打开文件
原型为:
int open(const char *path,int oflags)
int open(const char *path,int oflags,mode_t mode)
path:完整的文件路径
oflags:文件访问模式(只读,只写,可读写)
mode:设定文件访问权限
返回值:返回与文件关联的文件描述符,失败返回-1
注意这个描述符是唯一的,不与其他进程共享,文件对应的文件描述符并不是固定的。
2)write函数:写入文件
原型为:
size_t write(int fildes,const void *buf,size_t nbytes);
将buf缓冲区的前nbyts个字节,写入文件描述符files关联的文件中。
返回值:实际写入的字节数,失败返回-1
3)read函数:读取文件
原型为:
size_t read(int fildes,void *buf,szie_t nbytes);
读取文件描述符fildes关联的文件中前nbytes个字节,到buf缓冲区
返回值:实际读取的字节数,失败返回-1
4)close函数:关闭文件
原型为:
int close(int fildes);
返回值:成功返回0,失败返回-1
3.调用系统函数示例
从一个文件中复制数据到另一个文件,代码如下:
copy.c
#include<fcntl.h>
#include<stdlib.h>
int main()
{
char c='\0';
int in=-,out=-;
in=open("myfile.txt",O_RDONLY);
//以只写方式创建文件,若文件不存在则新建一个文件
//文件所有者具有读和写权限
out=open("myfile2.txt",O_WRONLY|O_CREAT,S_IRUSR|S_IWUSR);
while(read(in,&c,)==)
write(out,&c,);
close(in);
close(out);
return ;
}
输出结果:
4.标准I/O库
在标准I/O中,使用的是文件流,对应着底层的文件描述符。
文件流是一个指向FILE结构的指针。
I/O库的函数包含在头文件stdio.h中。
1)fopen函数:打开文件
原型为:
FILE* fopen(const char *filename,const char *mode);
与底层open函数类似
返回值:成功返回非空指针,失败返回NULL
2)fread函数:读取文件
原型为:
size_t fread(void *ptr,size_t size,size_t nitems,FILE *stream);
与底层read函数类似
从stream读取nitems个长度为size的数据到ptr指向的缓冲区
返回值:成功读取的字节数,失败返回-1
3)fwrite函数:写入文件
原型为:
size_t fwrite(const void *ptr,size_t size,size_t nitems,FILE *stream);
与底层write函数类似
从ptr指向的缓存区读取nitems个长度为size的数据,并把它们写到stream对应的文件中。
返回值:成功写入的字节数,失败返回-1
4)fclose函数:关闭文件
原型为:
int fclose(FILE *stream);
返回值:成功返回0,失败返回-1
5.标准I/O使用示例
与前例一样,从一个文件中复制数据到另一个文件,只是使用I/O库函数来实现,
代码如下:
#include<stdio.h>
#include<stdlib.h>
int main()
{
int c=;
FILE *pfin = NULL;
FILE *pfout = NULL;
pfin = fopen("myfile.txt","r");
pfout = fopen("myfile2.txt","w");
while(fread(&c,sizeof(char),,pfin))
fwrite(&c,sizeof(char),,pfout);
fclose(pfin);
fclose(pfout);
return ;
}
输出结果:
程序中的读和写数据可以用库中的其他函数来代替,如fget,fputc等。
6.文件描述符和文件流
一般不要混合使用底层输入输出与高层文件流操作。
调用fileno函数,可以获得文件流使用的底层文件描述符
原型为:int fileno(FILE *stream);
调用fdopen函数,可以在一个已经打开的文件描述符上创建一个新的文件流
原型为:FILE* fdopen(int fildes,const char* mode);
Linux下编程时,一般使用系统调用,而不使用I/O库。
有些操作必须使用系统调用,如创建文件读写锁等。
7.系统调用性能优化
我们使用time命令测试系统调用和I/O库两种方法的运行时间,
结果如下所示:
可以看出,系统调用的效率要明显低于I/O库,这是为什么呢?
因为系统调用时,Linux必须从运行用户代码切换到内核代码,然后再返回用户代码。
而I/O库函数会自动在数据满足数据块长度时,才调用底层系统函数。
系统调用代码优化如下,每次一次性读取和写入N个字节,减少系统调用次数。
copyopt.c
#include<fcntl.h>
#include<stdlib.h>
int main()
{
char buffer[];
int in=-,out=-;
int nread=;
in=open("myfile.txt",O_RDONLY);
//以只写方式创建文件,若文件不存在则新建一个文件
//文件所有者具有读和写权限
out=open("myfile2.txt",O_WRONLY|O_CREAT,S_IRUSR|S_IWUSR);
while(nread=read(in,buffer,sizeof(buffer))>)
write(out,buffer,nread);
close(in);
close(out);
return ;
}
使用time测试输出如下:
可以看出,性能得到了提高,甚至超过了使用I/O的性能。