文件IO

2021-05-31 12:46:14 星期一

文件描述符:是有限资源

文件描述符 POSIX名称 用途 stdio流
0 STDIN_FILENO 标准输入 stdin
1 STDOUT_FILENO 标准输出 stdout
2 STDERR_FILENO 标准错误 stderr

基础IO

open

#include <fcntl.h>

int open(const char *path, int oflag, mode_t mode);
  • path文件的路径和名称
  • flag文件的打开模式,组合使用时需要使用位运算或
    • 基本模式

      • 只读 只写 读写
        O_RDONLY O_WRONLY O_RDWR
        00 01 02
    • 附加模式(只列常用)
      • O_APPEND:总是在文件末尾添加数据
      • O_EXCL:配合O_CREAT标志
        • 表明如果文件存在则不会打开文件,并使open调用失败,否则能够创建并打开文件。这样确保了调用open()的进程即为创建文件的进程.
        • 同时不允许path是符号链接
      • O_CREAT:没有文件存在时会创建文件,需要mode参数指明文件权限,一共9个
      • O_TRUNC:如果文件存在为普通文件且该进程对改文件有写权限,则清空文件内容
  • mode
    • S_IRUSR:文件所有者有读权限
    • S_IWUSR:文件所有者有写权限
    • S_IXUSR:文件所有者有执行权限
    • S_IRGRP:同组用户有读权限
    • S_IWGRP:同组用户有写权限
    • S_IXGRP:同组用户有执行权限
    • S_IROTH:其他用户有读权限
    • S_IWOTH:其他用户有写权限
    • S_IXOTH:其他用户有执行权限
  • 返回文件描述符fd或者-1

错误

creat

旧版使用,新版都用open进行创建文件

read

open返回的fd文件描述符中读取bytes字节的数据到buf中,返回实际读取的字节数,不成功返回-1

#include <unistd.h>

ssize_t read(int fd, void *buf, size_t bytes);
  • read普通文件:调用成功返回实际读取的字节数,遇到文件EOF时返回0,出现错误返回-1
  • read读取终端,遇到\n即返回

注意:read系统调用是逐字节读取的,所以无法遵守C语言中的字符串的规则。比如C语言中字符串以\00x0)作为结束,但是read认为这里的字节值是0x0并继续读下去.所以使用read读数据时,通常的方式是:每次读取数据,再将的实际读取到的size处设置成C语言认可的字符串结束符,即buffer[size] = '\0';

#define MAX_READ 16

char buffer[MAX_READ + 1];
ssize_t size;
size = read(fd, buffer, MAX_READ);
if (size == -1)
exit(0);
buffer[size] = '\0';
close(fd);

一个例子

该文件从终端读取一行(因为read读终端时以\n作为结束)字符并打印出来,同时打印每一个字符

#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <string.h> #define MAX_READ 16 int main(int argc, char **argv)
{
char buffer[MAX_READ + 1];
memset(buffer, 0x23, sizeof(buffer)); // fill with #
buffer[MAX_READ] = '\0'; ssize_t numRead, i;
numRead = read(STDIN_FILENO, buffer, MAX_READ);
if (numRead == -1)
perror("read"); // buffer[numRead] = '\0';
printf("the input data was:%s\n", buffer);
for (i = 0; buffer[i] != '\0' && i < MAX_READ; i++)
printf("%ld->%x\n", i, buffer[i]);
return 0;
}
$ ./4.4
abcd
the input data was:abcd
############
0->61
1->62
2->63
3->64
4->a
5->23
6->23
...
16->0

由此可见read只是以二进制的形式照搬数据,并不对数据进行处理,因此,对数据的处理留给了程序员

write

bytes字节的buf数据写到open返回的fd文件描述符所指的文件中,返回实际写的字节数,不成功返回-1写入已打开的文件。调用成功并不代表已经写入磁盘,可能先进入缓存(这样减少磁盘活动量、加快write调用)。

#include <unistd.h>

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

close

#include <unistd.h>

int close (int fd);

close函数也有错误处理,编程时也应该错误检查。

lseek

内核打开的文件时会记录文件偏移量,第一字节的偏移量为0,文件打开时,会将偏移量设置为0

十分重要:有时候读文件读不出来,可能就是因为文件偏移量在文件末尾处,这时候需要重置

#include <unistd.h>

off_t lseek(int fd, off_t offset, int whence);
  • fd文件描述符
  • offset表示偏移量
    • 正值为正向移动,向继续往下读的方向
  • wherece表示文件读写指针从哪里开始计数
    • SEEK_SET表示起始位置
    • SEEK_CUR表示当前位置
    • SEEK_END表示末尾位置的后一个字节(这里直接写数据的话是恰好和文件连接)
  • 返回新的文件偏移量或-1(执行失败)

文件空洞

从文件结尾后到新写入的数据间的空间,他不占用磁盘空间,直到写入了数据。这时文件的名义的大小可能比磁盘存储的总量大。具体的在14节

ls -l file        查看文件逻辑大小

du -c file     查看文件实际占用的存储块多少

od -c file     查看文件存储的内容

unlink删除

只是删除path到文件的一个链接,其文件对于的i-node减1,为0时,改文件才从磁盘删除。

#include <unistd.h>

int unlink(const char *path);

iotcl

文件和目录

目录是另一种文件,只是内容是包含的文件信息和目录信息。

链接

每一个文件对应一个inode,文件的链接数对应inode中的链接数,记录着这个文件的链接数值,即指向该inode的文件数,文件和inode是多对一的关系。本质上rm指令调用系统调用unlink函数,将这个文件的inode的链接数-1,为0时才真正删除

  • 硬链接

    • 相链接的文件总是同步
  • 软链接
    • 理解为Windows的快捷方式
$ touch a.txt
$ echo "hello" > a.txt $ ls -li
total 4
2104443 -rw-rw-r-- 1 dwr dwr 6 Apr 10 22:20 a.txt
# inode编号 文件权限 用户 组用户 不知道 创建月 日 时 文件名
$ ln a.txt a.txt.bak # 建立硬链接
$ ls -li
total 8
2104443 -rw-rw-r-- 2 dwr dwr 6 Apr 10 22:20 a.txt
2104443 -rw-rw-r-- 2 dwr dwr 6 Apr 10 22:20 a.txt.bak $ ln -s a.txt a.txt.s # 建立软链接
$ ls -li
total 8
2104443 -rw-rw-r-- 2 dwr dwr 6 Apr 10 22:20 a.txt
2104443 -rw-rw-r-- 2 dwr dwr 6 Apr 10 22:20 a.txt.bak
2099642 lrwxrwxrwx 1 dwr dwr 5 Apr 10 22:25 a.txt.s -> a.txt
#include <unistd.h>

int link(const char *__from, const char *__to);
int symlink(const char *__from, const char *__to);

错误打印

perror

<errno.h>

根据设置的errno值打印对应的错误信息,打印规则是先打印s中用户定义的错误输出,在打印系统调用错误的输出提示。一定要在系统调用之后紧跟打印,否则会被覆盖

void perror(const char *s);

strerror

将错误代码转换为字符串错误信息。

char *strerror(int errno);

原子IO

fcntl

#include <fcntl.h>

int fcntl(int __fd, int __cmd, ...)

文件IO缓冲

这里是Unix系统编程手册第13章内容,不全待完善

read()write()系统调用在操作磁盘文件时不会直接发起磁盘访问,而是仅仅在用户空间缓冲区与内核缓冲区高速缓存(kernel buffer cache)之间复制数据。write()在后续某个时刻,内核会将其缓冲区中的数据写入(刷新至)磁盘。

Linux 内核对缓冲区高速缓存的大小没有固定上限。内核会分配尽可能多的缓冲区高速缓存页,而仅受限于两个因素;可用的物理内存总量,以及出于其他目的对物理内存的需求(例如,需要将正在运行进程的文本和数据页保留在物理内存中)。若可用内存不足,则内核会将

解释一下书中对IO系统调用的实验:

  • 总用时=CPU用时+磁盘读写用时
  • CPU用时=用户CPU用时(用户模式下执行的代码)+系统CPU用时(内核模式(系统调用和数据在用户和内核模式下传输)下执行的代码)

stdio的缓冲

C语言中IO函数可以理解为系统IO调用+数据缓冲,免于编写者自己处理对数据的缓冲

int setvbuf(FILE *stream, char *buf, int modes, size_t n)

控制stdio库函数的缓冲形式,需要最先调用,之后的stdio操作才有效

  • stream

    • 表示配置缓冲的文件流
  • buf
    • 不为空则使用size大小作为缓冲区(这个buf空间应该是堆内存上,避免函数调用和返回对栈进行修改)
    • 为空则自动分配(根据mode选择是否分配)size大小空间
  • modes
    • _IONBF:not,不缓冲,每一次调用stdio函数都立即调用系统调用
    • _IOLBF:line,行缓冲,遇到换行符或缓冲区满则调用系统调用,指向终端设备的流默认使用该模式
    • _IOFBF:file,全缓冲,指向磁盘的流默认使用该模式
  • 出错返回非0,成功返回0

Linux C 文件IO的更多相关文章

  1. linux中文件IO

    一. linux常用文件IO接口 1.1. 文件描述符 1.1.1. 文件描述符的本质是一个数字,这个数字本质上是进程表中文件描述符表的一个表项,进程通过文件描述符作为index去索引查表得到文件表指 ...

  2. 漫谈linux之文件IO篇(SSD写性能和机械硬盘差不多,读是4到10倍)

    前同事的文章,觉得写得很清晰,收藏了. http://blog.chinaunix.net/uid-27105712-id-3270102.html 在Linux 开发中,有几个关系到性能的东西,技术 ...

  3. linux下 文件IO 相关

    linux下操作文件或设备,需要一个文件描述符 file descriptor,fd 来引用.fd是一个非负整数,实际上是一个索引值,指向文件的记录表,对文件的操作都需要fd.默认的几个:标准输入流 ...

  4. 【Linux】文件IO --- sync、fsync、fdatesync

    在使用write函数向文件中写入数据的时候,并不是在调用了函数以后就直接把数据写入磁盘:操作系统在内核中设置了一块专门的缓冲区,数据会先被写入到内核的缓冲区中,等到缓冲区满了或者系统需要重新利用缓冲区 ...

  5. imx6用文件io操作gpio

    具体请参考: http://blog.csdn.net/u014213012/article/details/53140781 这里要注意的是: 要让linux支持文件io方式操作gpio,首先驱动必 ...

  6. 转 漫谈linux文件IO

    在Linux 开发中,有几个关系到性能的东西,技术人员非常关注:进程,CPU,MEM,网络IO,磁盘IO.本篇文件打算详细全面,深入浅出.剖析文件IO的细节.从多个角度探索如何提高IO性能.本文尽量用 ...

  7. 漫谈linux文件IO

    在Linux 开发中,有几个关系到性能的东西,技术人员非常关注:进程,CPU,MEM,网络IO,磁盘IO.本篇文件打算详细全面,深入浅出.剖析文件IO的细节.从多个角度探索如何提高IO性能.本文尽量用 ...

  8. Linux中的IO复用接口简介(文件监视?)

    I/O复用是Linux中的I/O模型之一.所谓I/O复用,指的是进程预先告诉内核,使得内核一旦发现进程指定的一个或多个I/O条件就绪,就通知进程进行处理,从而不会在单个I/O上导致阻塞. 在Linux ...

  9. Linux下用文件IO的方式操作GPIO(/sys/class/gpio)(转)

    通过sysfs方式控制GPIO,先访问/sys/class/gpio目录,向export文件写入GPIO编号,使得该GPIO的操作接口从内核空间暴露到用户空间,GPIO的操作接口包括direction ...

随机推荐

  1. Spring Boot demo系列(二):简单三层架构Web应用

    2021.2.24 更新 1 概述 这是Spring Boot的第二个Demo,一个只有三层架构的极简Web应用,持久层使用的是MyBatis. 2 架构 一个最简单的Spring Boot Web应 ...

  2. 9. resultMap 结果映射集

    @Data public class CreditCard extends BankCard { /** * 消费额度 */ private String creditLine; } @Data pu ...

  3. Neo4j/cypher学习笔记与学习建议

    简介 本笔记的主要内容是 cypher 查询语言的编写与使用. 笔记主要整理自w3cschool上的neo4j教程以及Neo4j中文网所提供的cypher中文文档,此外还包括少量从其他个人博客与官方手 ...

  4. 刨死你系列——手撕ArrayList

    不多BB,直接上代码: public class MyArrayList { //创建数组对象 private Object[] elements; //已使用数组长度 private int siz ...

  5. JUC包的线程池详解

    为什么要使用线程池 创建/销毁线程需要消耗系统资源,线程池可以复用已创建的线程. 控制并发的数量.并发数量过多,可能会导致资源消耗过多,从而造成服务器崩溃.(主要原因) 可以对线程做统一管理. JUC ...

  6. 幻读:听说有人认为我是被MVCC干掉的

    @ 目录 前言 系列文章 一.我是谁? 二.为什么有人会认为我是被MVCC干掉的 三.我真的是被MVCC解决的? 四.再聊当前读.快照读 当前读 快照读 五.告诉你们吧!当前读的情况下我是被next- ...

  7. Ionic5路由跳转传值复用

    1. 路由技术 ( 详细记录 ) 是笔记不是博文,觉得写的不够详细的可以使用Ctrl + W组合键 路由跳转页面 1. HTML 中使用 routerLink 属性路由进行跳转,传值时使用 query ...

  8. java面试一日一题:讲下mysql中的索引

    问题:请讲下mysql中的索引 分析:mysql中有很多索引,要对对这些索引有所掌握,还要弄清楚每种索引的本质? 回答要点: 主要从以下几点去考虑 1.索引的本质是什么 2.mysql的索引分类: 3 ...

  9. 我的自定义多交互live2d折腾经历

    在@m0d1 大佬的督促(?)下有了这篇复盘.不过因为可能很多地方讲得不全面+理解不够深入,故不打算把这篇当成是教程/指南,那就算是一个指北吧= = (划重点:不是教程!不是教程!不是教程! 省流简介 ...

  10. 【ElasticSearch】ElasticSearch集群扫盲

    Cluster 集群 ⼀个 Elasticsearch 集群由⼀个或多个节点(Node)组成,每个集群都有⼀个共同的集群名称作为标识.   Node节点 ⼀个 Elasticsearch 实例即⼀个 ...