Linux文件操作

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Linux文件操作相关的知识,希望对你有一定的参考价值。

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=-1,out=-1;
    in=open("myfile.txt",O_RDONLY);
    //以只写方式创建文件,若文件不存在则新建一个文件
    //文件所有者具有读和写权限
    out=open("myfile2.txt",O_WRONLY|O_CREAT,S_IRUSR|S_IWUSR);
    while(read(in,&c,1)==1)
        write(out,&c,1);
    close(in);
    close(out);
    return 0;
}

输出结果:

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=0;
    FILE *pfin = NULL;
    FILE *pfout = NULL;
    pfin = fopen("myfile.txt","r");
    pfout = fopen("myfile2.txt","w");
    while(fread(&c,sizeof(char),1,pfin))
        fwrite(&c,sizeof(char),1,pfout);
    fclose(pfin);
    fclose(pfout);
    return 0;
}

输出结果:

程序中的读和写数据可以用库中的其他函数来代替,如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[1024];
    int in=-1,out=-1;
    int nread=0;
    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))>0)
        write(out,buffer,nread);
    close(in);
    close(out);
    return 0;
}

使用time测试输出如下:

可以看出,性能得到了提高,甚至超过了使用I/O的性能。

以上是关于Linux文件操作的主要内容,如果未能解决你的问题,请参考以下文章

VSCode自定义代码片段15——git命令操作一个完整流程

VSCode自定义代码片段15——git命令操作一个完整流程

VSCode 如何操作用户自定义代码片段(快捷键)

[linux][c/c++]代码片段01

[linux][c/c++]代码片段02

LINUX PID 1和SYSTEMD PID 0 是内核的一部分,主要用于内进换页,内核初始化的最后一步就是启动 init 进程。这个进程是系统的第一个进程,PID 为 1,又叫超级进程(代码片段