当前位置: 首页 > news >正文

【Linux】简单设计libc库

📝前言:

经过之间两篇文章,【Linux】基础IO(一)和【Linux】基础IO(二)的学些,我们对文件的基础IO已经有了一定的理解。
这篇文章我们来简单设计一下libc库,来复习一下文件基础IO的知识

🎬个人简介:努力学习ing
📋个人专栏:Linux
🎀CSDN主页 愚润求学
🌄其他专栏:C++学习笔记,C语言入门基础,python入门基础,C++刷题专栏


目录

  • 一,综述
  • 二,Linux内核的_IO_FILE
  • 三,mystdio.h的编写
    • 实现
  • 四,mystdio.c的编写
    • 接口介绍
    • 实现
  • 五,test.c测试

一,综述

目标:用我们的文件实现文件的写入操作

本文主要完成三个文件的编写

  • mystdio.h:头文件,主要用于声明函数,提供接口
  • mystdio.c:源文件,用于实现头文件中声明的函数和全局变量
  • test.c:测试文件

二,Linux内核的_IO_FILE

如图,这就是C库里面,管理文件的结构体
在这里插入图片描述

三,mystdio.h的编写

  • C语言库的FILE结构体,就是struct _IO_FILE FILE结构体typedef来的
  • 目标封装:
    • IO_FILE结构体(C语言的结构体),核心参数
      • int flag :文件的刷新方式
      • int fileno:文件描述符
      • char outbuffer[SIZE]:输出缓冲区(语言层)
      • int cap:缓冲区的总容量
      • int size:缓冲区已经使用的字节数
    • myopenmywritemyfflushmyclose操作,模拟C语言里面库函数的调用

实现

  1 #pragma once2 3 #define MAXSIZE 10244 5 #define FLUSH_N 06 #define FLUSH_L 17 #define FLUSH_F 28 9 struct IO_FILE10 {11     int flag;12     int fileno;13     char outbuffer[MAXSIZE];14     int cap;15     int size;16 };17 18 typedef struct IO_FILE MYFILE;19 20 MYFILE* myfopen(const char* pathname, const char* mode);21 int myfwrite(const void* ptr, int num, MYFILE* stream); //void* ptr这使得 mfwrite 函数能够接受任意类型的数据缓冲区。                                                                                         22 void myfflush(MYFILE* stream);23 void myfclose(MYFILE* stream);                      

四,mystdio.c的编写

接口介绍

memcpy

  • 原型:void* memcpy(void* dest, const void* src, size_t n);
    • dest:目标内存区域的起始地址(需确保有足够空间)
    • src:源内存区域的起始地址(只读)
    • n:要复制的字节数(size_t 类型,通常由 sizeof 或手动计算得到)

fsync

  • 传入文件描述符,把对应的内核文件缓冲区刷新到外设(我们不用管如何刷新的,只要知道能刷新就行了)

openclose等文件的系统调用不过多赘述,如果不懂的可以这篇文章:【Linux】基础IO(一)

实现

  1 #include "mystdio.h"2 #include <string.h>3 #include <stdlib.h>4 #include <sys/stat.h>5 #include <sys/types.h>6 #include <fcntl.h>7 #include <unistd.h>8 9 MYFILE* myfopen(const char* pathname, const char* mode)10 {11     // opne时,如果对不存在的文件,那就要创建文件并且初始化FILE结构体12     int fd = -1; // 代表 open失败13     // 对于不同的打开方式,我们主要通过更新它的 FILE 来体现文件不同的状态14     if(strcmp(mode, "r") == 0)15     {16         fd = open(pathname, O_RDONLY); // 封装系统调用17     }18     else if(strcmp(mode, "w") == 0)19     {20         fd = open(pathname, O_CREAT|O_WRONLY|O_TRUNC, 06660);21     }22     else if(strcmp(mode, "a") == 0)23     {24         fd = open(pathname, O_CREAT|O_WRONLY|O_APPEND, 06660);25     }26     if(fd < 0) return NULL; // 代表开失败了27     MYFILE* mf = (MYFILE*)malloc(sizeof(MYFILE)); // 申请MYFILE结构体,给新的文件28     if(!mf) // 分配失败29     {30         close(fd);31         return NULL;32     }33     mf -> fileno = fd;34     mf -> flag = FLUSH_L; // 我们默认用行刷新,简单一点35     mf -> size = 0;36     mf -> cap = MAXSIZE;37 38     return mf;39 }40 int myfwrite(const void* ptr, int num, MYFILE* stream)41 {42     // 语言层写到缓冲区,本质是拷贝                                                                                                                                                                          43     memcpy(stream->outbuffer + stream->size, ptr, num);44     stream -> size += num;45     if(stream -> size > 0 && stream -> flag == FLUSH_L && stream -> outbuffer[stream->size - 1] == '\n')46     {47         myfflush(stream);48     }49     return num;50 }51 void myfflush(MYFILE* stream)52 {53     if(stream->size > 0)54     {55         write(stream->fileno, stream->outbuffer, stream->size);56         // 刷新到外设57         fsync(stream->fileno);58         stream->size = 0;59     }60 }61 void myfclose(MYFILE* stream)62 {63     if(stream->size > 0)64     {65         myfflush(stream);66     }67     close(stream->fileno);68 }69 

五,test.c测试

测试往文件log.txt里面写入信息,看是否写入成功
代码:

  1 #include "mystdio.h"2 #include <stdio.h>3 #include <string.h>4 #include <unistd.h>5 6 int main()7 {8     MYFILE* fp = myfopen("./log.txt", "a");9     if(fp == NULL)10     {11         printf("打开失败");12         return 1;13     }14     int cnt = 3;                                                                                                                                                                                             15     while(cnt)16     {17         printf("write %d\n", cnt);18         char buffer[64];19         snprintf(buffer, sizeof(buffer),"hello message, number is : %d\n", cnt);20         cnt--;21         myfwrite(buffer, strlen(buffer), fp);22         myfflush(fp);23         sleep(1);24     }25     myfclose(fp);26     return 0;27 }

运行结果:
在这里插入图片描述


🌈我的分享也就到此结束啦🌈
要是我的分享也能对你的学习起到帮助,那简直是太酷啦!
若有不足,还请大家多多指正,我们一起学习交流!
📢公主,王子:点赞👍→收藏⭐→关注🔍
感谢大家的观看和支持!祝大家都能得偿所愿,天天开心!!!

相关文章:

  • python3:文件与异常
  • 武汉芯源半导体CW32L010在两轮车仪表的应用介绍
  • 从0到1上手Kafka:开启分布式消息处理之旅
  • Samtec Demo前沿 | 实时毫米波数据链路演示
  • 技术并不能产生一个好的产品
  • ETL背景介绍_1:数据孤岛仓库的介绍
  • 浅论3DGS溅射模型在VR眼镜上的应用
  • hivesql是什么数据库?
  • MySQL的Docker版本,部署在ubantu系统
  • 七、深入 Hive DDL:管理表、分区与洞察元数据
  • 自我奖励语言模型:突破人类反馈瓶颈
  • 智源联合南开大学开源Chinese-LiPS中文多模态语音识别数据集
  • 【datawhale组队学习】coze-ai-assistant TASK01
  • 电脑关机再开机会换IP吗?深入解析分配机制
  • 【HTML5】【AJAX的几种封装方法详解】
  • 【git】clone项目后续,github clone的网络配置,大型项目git log 输出txt,切换commit学习,goland远程,自存档
  • 逆强化学习IRL在医疗行为模式研究中的应用
  • 安卓工控一体机在智慧农业设备中的应用场景
  • Rimworld Mod教程 武器Weapon篇 近战章 第二讲:生物可用的近战来源
  • SpringBoot整合MQTT实战:基于EMQX实现双向设备通信(附源码)
  • 中国海警舰艇编队5月14日在我钓鱼岛领海巡航
  • 苹果或将于2027年推出由玻璃制成的曲面iPhone
  • 举牌代跳明码标价、留言不堪入目,未成年人擦边短视频成引流利器
  • 领证不用户口本,还需哪些材料?补领证件如何操作?七问七答
  • 总粉丝破亿!当网络大V遇见硬核科技,互联网时代如何书写上海故事?
  • 习近平会见古共中央第一书记、古巴国家主席迪亚斯-卡内尔