Linux 文件操作接口

在使用语言编写服务的时候不乏会遇到需要对文件进行操作的场景,Linux内核是用C语言写的,了解Linux之前先熟悉一下C语言文件操作接口,方便对比。

C语言文件操作接口

C语言文件描述

#ifndef _FILE_DEFINED
struct _iobuf {
    char *_ptr; //文件输入的下一个位置
    int _cnt; //当前缓冲区的相对位置
    char *_base; //指基础位置(即是文件的起始位置)
    int _flag; //文件标志
    int _file; //文件描述符id
    int _charbuf; //检查缓冲区状况,如果无缓冲区则不读取
    int _bufsiz; //文件缓冲区大小
    char *_tmpfname; //临时文件名
};
typedef struct _iobuf FILE;
#define _FILE_DEFINED
#endif

C语言对文件操作需要的数据都存在这样的数据结构里,C语言对文件操作时,用一个数据结构唯一标识一个文件流

fopen()

FILE* fopen(const char *path, const char *mode);

返回值为文件流结构体指针,当打开失败时返回NULL指针。

对文件操作前需先打开文件,打开文件使用接口fopen()。

参数:

path: 文件路径,可以是相对路径也可以是绝对路径(默认为进程打开时路径)

mode: 打开方式

模式 含义 文件不存在时
r 只读 报错
w 只写 创建文件
a 追加只写 创建文件
rb 二进制只读 报错
wb 二进制只写 创建文件
ab 二进制追加只写 创建文件
r+ 读写 报错
w+ 读写 创建文件
a+ 追加读写 创建文件
rb+ 二进制读写 报错
wb+ 二进制读写 创建文件
ab+ 二进制追加读写 创建文件

以上打开模式凡带'b'操作的打开文件时都会清空文件。

r模式打开文件

)

文件不存在时打开失败



文件存在时打开成功

w模式打开文件



自动创建需要打开的文件



打开后会清空文件

a模式打开文件

以a模式打开文件不存在时创建文件,存在时在文件末尾写入内容。

其他模式类似

fclose()

C语言程序员要养成用完即释放的好习惯尽量避免内存泄漏,fclose() 接口就是用来关闭文件流的。

int fclose(FILE *fp);

关闭接口参数简单,只需将需要关闭的文件流指针传入即可。

fwrite()

size_t fwrite(const void *ptr, size_t size, size_t number, FILE *stream);

参数:

ptr: 写入文件的内容

size: 写入单位数据大小(byte)

number: 写入数据总数

stream: 文件流

写入操作传参如图

fread()

size_t fread(void *ptr, size_t size, size_t number, FILE *stream);

参数类似fwrite()

ptr 为要读入文件内容的容器,必须提前开好空间,number 不得大于实际开好的空间

读文件操作如图

系统文件操作接口

文件描述符fd

文件描述符可以唯一标识该进程打开的流。

open()

int open(const char *pathname, int flags);
int open(const char *pathname, int flags, mode_t mode);

参数:

pathname: 文件路径,默认为进程创建时的路径

flags: 决定文件打开模式

mode: 创建文件时文件权限

理解标记位原理

计算机中所有数据都由比特位组成,一个整形有32个比特位,用一位作为标记,一个int类型可以携带32个标识,并且可以随意组合。

如下操作,想完成什么操作传入对应的标记即可。

open()的常用标记

标记 含义
O_RDONLY 只读
O_WRONLY 只写
O_RDWR 读写
O_APPEND 追加
O_CREAT 创建文件
O_TRUNC 清空文件

将只读标记和创建文件标记传入open()接口,创建文件如图。文件创建出来了,但可以看出它的权限是乱的,可见这个接口不足以像fopen() 一样打开文件。

使用另一个接口创建出来权限正常的文件,但权限还不是如我们所设想的设置什么权限码就创建什么权限文件。此问题与权限掩码umask有关

权限码与umask取反再按位与最后得到的权限码才是最终权限码,若需只在该程序重设umask 只需使用接口umask:

mode_t umask(mode_t mask);

参数mask为想要重设的权限掩码

现在所创建的文件权限就全如程序员所愿了。

write()

系统写文件操作接口

ssize_t write(int fd, const void *buf, size_t count);

操作与fwrite() 基本一致,参数buf 类型为void* ,count 为要写入内容大小(byte)。

写入成功,但如上打开方式还有一个隐患:

后写入的内容并不会在空文件写入,而是会在已有内容上进行覆盖,这样的操作可不像C语言的w操作,针对此问题可以再加一个标志O_TRUNC

如此才可和C语言的fopen() 的 w 操作一致

read()

ssize_t read(int fd, void *buf, size_t count);

read()参数与write() 基本一致,与fread() 操作相差不多,需要buf 提前开好空间

当文件不存在时:

如此,效果与fread() 一致

close()

使用完文件后需得关闭文件,尽量防止内存泄漏,使用close()接口

int close(int fd);

两种操作接口联系

操作系统设计时为了安全任何高级语言操作文件都不能绕过操作系统,用户对文件操作只能通过操作系统提供的接口,而Linux操作系统内核主要用C语言编写,提供的接口也是用C语言编写的,C语言用户可以直接使用。C语言操作文件也需通过操作系统提供的接口,因此C语言的文件操作接口是对系统文件接口进行封装后暴露出来的。

但既然已经有系统接口了为什么还要C语言文件操作接口,此问题原因有几个:

  1. 使用不方便(对比之前的fopen()open())
  2. C语言是跨平台语言,若使用系统文件操作接口,同一份代码移植到其他平台将会编译不通过,使用封装后的接口,可以完美解决这个问题

高级语言都是用自己的语言特性封装系统接口来操作文件

系统接口 语言接口
open(const char* path, O_WRONLY | O_CREAT | O_TRUNC) fopen(const char* path, "w")
open(const char* path, O_RDONLY) fopen(const char* path, "r")
open(const char* path, W_WRONLY | O_CREAT |O_APPEND) fopen(const char* path, "a")
close(int fd) fclose(FILE* fStream)
write(fd, const void* buf, size_t count) fwrite(const void* ptr, size_t size, size_t number, FILE* fStream)
read(fd, void* buf, size_t count) fread(void* ptr, size_t size, size_t number, FILE* fStream)

C语言对封装的接口跨平台解决办法

条件编译 + 穷举

Linux 文件操作接口的更多相关文章

  1. C/C++以及Linux文件操作备忘录

    目录 C文件操作 文件开关 文件读写 C++文件操作 Linux文件操作 打开 C文件操作 #include<stdio.h> stdin, stdout, stderr 文件开关 /* ...

  2. Linux文件操作的主要接口API及相关细节

    操作系统API: 1.API是一些函数,这些函数是由linux系统提供支持的,由应用层程序来使用,应用层程序通过调用API来调用操作系统中的各种功能,来干活 文件操作的一般步骤: 1.在linux系统 ...

  3. Linux 文件操作监控inotify功能及实现原理【转】

    转自:http://blog.chinaunix.net/uid-26585427-id-5012973.html 1. inotify主要功能 它是一个内核用于通知用户空间程序文件系统变化的机制. ...

  4. Linux 文件操作总结

    http://blog.163.com/he_junwei/blog/static/19793764620152592737741/ ioctl?? lseek?? 文件是linux中的一个重要概念. ...

  5. Linux文件操作学习总结【转载】

    本文转载自: http://blog.csdn.net/xiaoweibeibei/article/details/6556951 文件类型:普通文件(文本文件,二进制文件).目录文件.链接文件.设备 ...

  6. 关于文件的INode与Java中的文件操作接口

    本文由作者周梁伟授权网易云社区发布. 近日做的项目中涉及到多进程共同读写多个文件的问题,文件名和最后修改时间都是可能会被频繁修改的,因而识别文件的唯一性会产生相当的麻烦,于是专门再学习了一下文件系统对 ...

  7. Linux 文件操作——系统调用和标准I/O库

    一.什么是文件 在讲述文件操作之前,我们首先要知道什么是文件.看到这个问题你可能会感觉到可笑,因为对于用过计算机的人来说,文件是最简单不过的概念了,例如一个文本是一个文件,一个work文档是一个文件等 ...

  8. linux 文件操作与目录操作

    文件操作 使用命令 命令格式: 命令 [选项] [参数] [] 表示可选的 示例: ls -a /etc 常识命令 ls:查看指定目录的内容,不指定目录时查看当前工作目录 选项 说明 -a 显示所有文 ...

  9. Linux文件操作常用命令整理

    收集.整理日常系统管理或维护当中的,常用到的一些关于文件操作的命令或需求,后续会慢慢补充.完善! 查看.生成指定目录的目录树结构?   [root@DB-Server ~]#tree   #当前目录 ...

随机推荐

  1. 【原创】Python 二手车之家车辆档案数据爬虫

    本文仅供学习交流使用,如侵立删! 二手车之家车辆档案数据爬虫 先上效果图 环境 win10 python3.9 lxml.retrying.requests 需求分析 需求: 主要是需要车辆详情页中车 ...

  2. Java版的防抖(debounce)和节流(throttle)

    概念 防抖(debounce) 当持续触发事件时,一定时间段内没有再触发事件,事件处理函数才会执行一次,如果设定时间到来之前,又触发了事件,就重新开始延时. 防抖,即如果短时间内大量触发同一事件,都会 ...

  3. 解决国内不能访问github的问题

    问题 最近访问GitHub总是不稳定,经常连不上, 出各种错误(OpenSSL SSL_read: Connection was reset, errno 10054, Connection refu ...

  4. 一文搞懂 Python 的模块和包,在实战中的最佳实践

    最近公司有个项目,我需要写个小爬虫,将爬取到的数据进行统计分析.首先确定用 Python 写,其次不想用 Scrapy,因为要爬取的数据量和频率都不高,没必要上爬虫框架.于是,就自己搭了一个项目,通过 ...

  5. 历时2月,动态线程池 DynamicTp 发布里程碑版本 V1.0.8

    关于 DynamicTp DynamicTp 是一个基于配置中心实现的轻量级动态线程池管理工具,主要功能可以总结为动态调参.通知报警.运行监控.三方包线程池管理等几大类. 经过多个版本迭代,目前最新版 ...

  6. Mybatis的ResultMap与limit分页查询

    ResultMap主要解决的是:属性名和字段不一致 如果在pojo中设置的是一个名字,在数据库上又是另一个名字,那么查询出来的结果或者其他操作的结果就为null. //在pojo中 private S ...

  7. 面试突击78:@Autowired 和 @Resource 有什么区别?

    @Autowired 和 @Resource 都是 Spring/Spring Boot 项目中,用来进行依赖注入的注解.它们都提供了将依赖对象注入到当前对象的功能,但二者却有众多不同,并且这也是常见 ...

  8. 第九十篇:Vue 具名插槽

    好家伙 1.什么是具名插槽? 来简单理解一下, 具有自己名字的插槽,就是具名插槽 我们来尝试使用一下具名插槽: 在Article.vue组件中: <template> <div cl ...

  9. 为什么Index Only Scan却还需要访问表

    在实际SQL优化工作中,我们经常会发现SQL 执行计划明明是 "Index Only Scan",但执行计划后面却有 "Heap Fetches: x" ,也就 ...

  10. KingbaseFlySync 需要对外开放的端口

    Oracle到kes双轨灾备场景 源:Oracle rac 11g 目标端:kes v8r6c4b21 源.目标.管控服务器IP 需要开放端口 为什么源和目标需要互相开放数据库端口:因为在双轨运行的方 ...