Linux系统中提供了系统调用函数open()和close()用于打开和关闭一个存在的文件
int open(const char *pathname,int flags)
int open(const char *pathname,int flags,mode_t mode)
int open(const char *pathname,mode_t mode)
其中flag代表文件的打开方式
O_RDONLY: 以只读方式打开文件
O_WRONLY:以只写的方式打开文件
O_RDWR:以读写的方式打开文件
O_CREAT:若打开的文件不存在,则创建该文件
O_EXCL:如果打开文件是设置了O_CREAT,但是该文件存在,则导致调用失败
O_TRUNC:如果以只写或只读方式打开一个已存在的文件,将该文件截至0
O_APPEND:追加的方式打开文件
O_NONBLOCK:用于非堵塞接口i/o
O_NODELAY
O_SYNC:当数据被写入外存或者其他设备后,操作才返回。
其中mode是打开的权限
S_IRWXU 00700 设置文件所有者的读,写,执行权限
S_IRWXG 00070 设置文件所在用户组的读,写,执行权限
S_IRWXO 00007 设置其他用户的读,写,执行权限
S_IRUSR 00400 设置文件所有者的读权限
S_IWUSR 00200 设置文件所有的写权限
S_IXUSR 00100 设置文件所有者的执行权限
S_IRGRP 00040 设置用户组的读权限
S_IWGRP 00020 设置用户组的写权限
S_IXGRP 00010 设置用户组的执行权限
S_IROTH 00004 设置其他用户的读权限
S_IWOTH 00002 设置其他用户的写权限
S_IXOTH 00001 设置其他用户的执行权限
文件的读操作:
ssize_t
read(int fd,void *buf,size_t count);
fd代表文件描述符,buf代表读取的数据存放的buf指针所指向的缓冲区,count代表读取数据的字节数
函数调用成功,返回为读取的字节数,否则返回-1
文件读和写的例子
void
open_and_read_file()
{
int
fd,n;
char
buf[100];
char
*path="/home/zhf/zhf/c_prj/c_test.txt";
fd=open(path,O_RDWR);
n=read(fd,buf,20);
printf("the
content is %s\n",buf);
}
在这里文件都是从头开始读的,那么如果我想从某个位置比如第N个字节开始读取的时候该如何操作呢,这里就需要用到文件定位函数lseek
off_t
lseek(int fildes,off_t offset,int whence);\
fildes是文件描述符
offset是偏移量
whence代表用于偏移时的相对位置,可以取如下的几个值
SEEK_SET:
从文件的开头位置计算偏移量
SEEK_CUR:
从当前的位置开始计算偏移量
SEEK_END:
从文件的结尾开始计算偏移量
函数修改如下:
首先采用lseek函数将文件定位到从第11个字节开始,然后read将从第11个字节开始读取。
对于写操作也是一样的。
void
open_and_read_file()
{
int
fd,n;
char
buf[100];
char
*path="/home/zhf/zhf/c_prj/c_test.txt";
int
i;
fd=open(path,O_RDWR);
i=lseek(fd,11,SEEK_SET);
n=read(fd,buf,20);
printf("the
content is %s\n",buf);
close(fd);
}
一个完整的创建,读,写文件的例子:
#include <fcntl.h>
#include <stdio.h>
#include <unistd.h>
void write_file(int fd){
char buf[]="abcde\n";
write(fd,buf,sizeof(buf));
close(fd);
}
void read_file(){
int fd;
char *path="/home/zhf/test1.txt";
char result[20];
fd=open(path,O_RDONLY);
read(fd,result,10);
printf("The content is %s",result);
close(fd);
}
int create_file(){
int fd;
char *path="/home/zhf/test1.txt";
fd=creat(path,00777);
return fd;
}
前面介绍的操作都是基于文件描述符的文件操作,下面来介绍基于流的文件操作。两者有什么区别呢:
缓冲文件系统的特点是:在内存开辟一个“缓冲区”,为程序中的每一个文件使用,当执行读文件的操作时,从磁盘文件将数据先读入内存“缓冲区”,
装满后再从内存“缓冲区”依此读入接收的变量。执行写文件的操作时,先将数据写入内存“缓冲区”,待内存“缓冲区”装满后再写入文件。由此可以看出,内存
“缓冲区”的大小,影响着实际操作外存的次数,内存“缓冲区”越大,则操作外存的次数就少,执行速度就快、效率高。一般来说,文件“缓冲区”的大小随机器
而定。主要有fopen,
fclose, fread, fwrite, fgetc, fgets, fputc, fputs, freopen, fseek,
ftell,
rewind等
2.非缓冲文件系统
缓冲文件系统是借助文件结构体指针来对文件进行管理,通过文件指针来对文件进行访问,既可以读写字符、字符串、格式化数据,也可以读写二进制数
据。非缓冲文件系统依赖于操作系统,通过操作系统的功能对文件进行读写,是系统级的输入输出,它不设文件结构体指针,只能读写二进制文件,但效率高、速度
快,由于ANSI标准不再包括非缓冲文件系统,因此建议大家最好不要选择它。本书只作简单介绍。open,
close, read, write, getc, getchar, putc, putchar 等。
open
是系统调用
返回的是文件句柄,文件的句柄是文件在文件描述副表里的索引,fopen是C的库函数,返回的是一个指向文件结构的指针。
linux中的系统函数是open,fopen是其封装函数。fopen可以移植,open不能
来看fopen函数:FILE
*fopen(const char *path,const char *mode)
path代表文件名路径
mode代表文件的打开方式。有
r,r+,w,w+,a,a+等方式
个字节
int
fgetc(FILE *stream). 如果出错或者读到文件末尾,则返回EOF
个文件
int
fputc(int c,FILE *stream);
实例代码:
void
fopen_file()
{
int
i;
char
ch;
FILE
*fp;
char
*path="/home/zhf/zhf/c_prj/c_test.txt";
fp=fopen(path,"r");
for(i=0;i<5;i++)
{
if
((ch=fgetc(fp)) != EOF)
{
printf("%c",ch);
}
}
}
fputc和fgetc函数只能一次性的读取或者写入一个字节,如果想读取或者写入一个字符串则需要用到fgets()和fputs()函数。
Char
*fgets(char *s,int size, FILE *stream);
读取到’\n’转移字符为结束,并在该行末尾添加一个’\0’组成完成的字符串。在size字节范围内没有读到’\n’结束符,则添加一个’\0’
int
fputs(const char *s, FILE *stream);
void
fopen_file()
{
int
i;
char
ch;
char
*str;
FILE
*fp;
char
*path="/home/zhf/zhf/c_prj/c_test.txt";
char
result[100];
fp=fopen(path,"r");
str=fgets(result,20,fp);
printf("%s\n",str);
fclose(fp);
}
同样的基于流操作的也有文件的定位操作函数分别是fseek函数和rewind函数
int
fseek(FILE *stream,long offset,int whence)
这里offset表示移动的字节数,要求位移量是long型数据。以便在文件长度大于64Kb不会出错
whence和前面介绍的一样。
rewind函数的作用是使位置指针重新返回文件的开头,函数没有返回值。