基础I/O

 基础IO:

  c库文件IO操作接口:(详细查看c语言中的文件操作函数总结:https://www.cnblogs.com/cuckoo-/p/10560640.html)

    fopen  打开文件  fclose   关闭文件  fread  读取文件  fwrite  写入文件  

    fseek  移动跳转当前读取/写入位置  

    printf    格式化字符串输出到终端  

    printf("%s-%s-%s-%d\n", "bite", "good", 666);

    sprintf  格式化字符串,将格式化后的字符串写入一个buff中

    sprintf(buff, "%s-%s-%s-%d\n", "bite", "good", 666);

    fprintf   格式化字符串,将格式化后的字符串写入文件中

    fprintf(stdout, "%s-%s-%s-%d\n", "bite", "good", 666);

    //stdout 标准输出——显示——linux下一切皆文件

  系统调用I/O函数:

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

      pathname:要打开的文件名称

      flags:标志选项

        必选项:(这三个选项相互冲突,只能选择其中一个)

          O_RDONLY  只读

          O_WRONLY  只写

          O_RDWR   读写

        非必选项:

          O_CREAT  若文件存在打开,不存在则创建

          O_EXCL  与O_CREAT同时用时,若文件存在则报错

          O_TRUNC  若文件存在则将文件长度截断为0

          O_APPEND  追加

      mode:若文件不存在需要创建的时候,用于指定创建的文件权限

      返回值:

        成功:非负整数(文件描述符,后续操作都通过描述符完成)

        失败:-1

  

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

      从buf中想fd所代表的文件写入count个字节的数据

      返回值:返回实际的写入长度,出错返回-1

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

      从fd所代表的文件中读取count字节长度的数据放到buf中

      返回实际读取的数据长度,出错返回-1

    lseek(fd, 0, SEEK_SET)

      跳转fd所代表的文件读写位置至起始位置的0偏移量处

  库函数和系统调用接口之间的关系:上下级调用关系

    库函数时对系统调用的一层封装  

 

  文件描述符:
    定义:返回一个数字,通过数字来管理文件
    进程通过struct file 结构体来描述打开的文件—使用了struct file* fd——array[]

    文件描述符就是这个数组的下标,操作系统通过file结构体描述文件
    并且将指针添加进入fd_array中,向用户返回这个文件描述信息在数组中的下标

    分配规则:
      最小未使用原则:默认从3开始,因为一个程序在运行时会默认打开3个文件

  文件描述符和文件流指针的关系:

         标准输入  标准输出  标准错误输出  

文件流指针    stdin    stdout    sterr 

文件描述符    0       1       2   

    文件指针用于库函数操作IO

    文件描述符用于系统调用接口操作IO

    文件流指针结构体中包含了一个成员就是我们的文件描述符

    文件流指针中还定义了一个缓冲区,我们所说的换行刷新缓冲区,实际就是刷新的这个缓冲区,这

个缓冲区用于将短小数据组成大数据一次性写入文件,这样可以提高效率(换行刷新缓冲区,只针对标准输

出),而系统调用则没有这个缓冲区。

   进程对文件的管理方式:先描述,在组织管理

    使用结构体来描述:

      文件标识符(inode结点号)  文件名称

重定向:
  改变数据流向,将写入指定文件的数据,改变之后写入另一个文件
  重定向值得是描述符的重定向,因为描述符并没有改变,改变的是描述符所对应的文件信息

  1.标准输出重定向:

    原本标准输入读取数据重新定向为从其他文件读取数据

    将原来要输出到标准输出的内容,重定向后输出到指定的其他文件

    <:标准输入重定向

    >:标准输出重定向  清空原有内容后添加新内容

    >>:标准输出重定向  向原有内容下方添加新内容


  dup2(int oldfd, int newfd)
  功能:让newfd指向oldfd所指向的文件;
  如果newfd本身已经打开了文件,则关闭原先打开的文件
  则nwefd和oldfd操作的都是oldfd所指向的文件

文件系统的理解: 

  文件系统(linux下ext2为例):
  磁盘上的文件管理系统(每一个磁盘分区上都会有一个文件管理系统,除了交换分区)

  分块管理:
  数据块区域
  inode结点区域
    inode结点:包含文件大小;权限;时间;占用块数,链接;文件数据存储地址;
  data_bitmap区域
    记录inode结点哪些是空闲的或者已经使用的
  超级块区域
    文件信息的统筹,以上分块的所有信息

  目录文件:
    记录了目录下文件的信息(文件名+inode结点号)——目录项
  文件存储的过程:
    通过inode_bitmap在inode区域获取空闲的inode结点,通过data_bitamap获取空闲数据块,在inode结点中记录文件
    信息以及数据快位置,并且将文件数据写入到数据块中。将自己的目录项信息添加到所在文件中

  文件读取过程:
    在目录项中通过文件名获取文件inode结点号(文件唯一)通过inode结点号在inode区域找inode区域找到inode结点,
    通过inode结点中的数据块地址信息,在指定数据块读取数据。

  查看inode结点大小:
    sudo dumpe2fs -h /dev/sda1 | grep "Inode size"

  软链接/硬链接(操作一个软/硬链接可以改变原文件)
    ln -s 原文件 创建的软链文件
    是一个独立的文件(像快捷方式)

    ln -h 原文件 创建的硬链文件
    一个文件的名字(目录项),与源文件公用一个inode结点

    ls -i (查看文件inode结点号)
    创建的硬链接文件的结点号与原文件相同
    软链中存储的是文件的路径——针对目录项

    软/硬链接的区别:
      删除源文件,软链接文件将失效;硬链接无影响,链接数-1(一个文件的链接数为0 时才是真正的删除了)
      软链接可以跨分区建立,硬链接不可以
      软链接可以对目录创建,硬链接不可以
      (软链接的操作可以跨分区,硬链接不可以)
  动态库/静态库

    动态库命名:lib前缀 .so是后缀 中间是库名称
    -fPIC //产生位置无关代码(课后调研,查找了解)
    1.gcc -fPIC -c child.c -o child.o
    2.gcc --share child.o -o libmychild.so(生成动态库/共享库)

  静态库命名:lib前缀 .a是后缀 中间是库名称
    gcc -c child.c -o child.o
    ar -cr libmychild.a child.o
    -c 创建静态库
    -r 模块替换

  链接库:
    gcc main.c -o main -lmychild
    -l 指定要链接的库名称(只取库名称)
    库文件有默认的搜索路径:lib/lib64/usr/lib64...

    链接时报错,没有找到库,原因:有默认搜索路径
    1.将库文件放入指定路径下 //根目录下lib64
    2.设置/声明环境变量:
      库文件的链接搜索路径:LIBRARY_PATH = .
      库文件的运行加载路径:LD_LIBRARY_PATH = .export
    3.在gcc生成可执行程序时,直接指定库的搜索路径
      gcc main.c -o main -L . -lmychild

      gcc -L //指定库的链接搜索路径

      gcc -I //指定头文件的搜索路径

    想使用第三方静态库,不能用 gcc 的 -static 选项,因为 -static 选项是生成静态链接可以执行
    程序,所有的库都是使用静态库(但我们只希望这个第三方库使用静态库,而不是所有),因此我们的做法
    是将第三方静态库拷贝到指定路径下,使用-L选项指定库的链接搜索路径那么这时候链接的就是静态库。

posted on 2019-06-24 21:50  The_Ocean  阅读(127)  评论(0编辑  收藏  举报

导航