Linux——缓冲区与实现C库的fopen,fwrite,fclose

目录

一.缓冲区

1缓冲区的概念

2.缓冲区存在的意义

3.缓冲区刷新策略   

4.什么是刷新?

C语言的缓冲区在哪里?

​编辑

仿写C库里的fopen,fclose,fwrite。

mystdio.h

mystdio.c

main.c(向文件中写入20次msg)


一.缓冲区

1缓冲区的概念

缓冲区的本质就是一段内存

2.缓冲区存在的意义

提高使用者的效率

同时因为缓冲区的存在也提高了操作系统的效率

举例一个例子:
 

假如你在云南要给你北京的朋友寄东西。方法一:你可以亲自己去北京把东西交给他,方法二:把东西给快递站,让它帮你送给他。

这里虽然整个事件持续的事件基本一样,但是对于你来说方法二比方法一效率要高的多,因为你只需要把东西交给快递站,对于你来说就结束了,接下来就可以做其他事情了。而快递站,也会根据你的需求以及自身制定发送快递的策略,不会收到一个快递就发送。

这里:你就是正在执行的进程,快递站是缓冲区。朋友就是文件,快递发送策略:缓冲区刷新策略


3.缓冲区刷新策略   

     1,无缓冲(立即刷新)--fflush()

   2,行缓存(行刷新)--遇到换行就刷新

一般对于显示器文件,采用行刷新

   3,全缓冲(缓冲区满了,再刷新)

磁盘文件采用全缓冲

特殊情况:

1.强制刷新

2.进程退出时,一般要进行刷新缓冲区(属于强制刷新的一种特殊情况)

一个例子:(同一份代码不同结果)

#include <stdio.h>
#include <string.h>
#include <unistd.h>int main()
{fprintf(stdout, "C: hello fprintf\n");printf("C: hello printf\n");fputs("C: hello fputs\n", stdout);const char *str = "system call: hello write\n";write(1, str, strlen(str));fork(); // 注意fork的位置!return 0;
}

可以看出,当重定向到log.txt文件时,除了系统函数write之外,都重复了两次。且write打印的内容在最前面

解释:

1.当我们直接向显示器打印时,显示器文件的刷新方式是行刷新!而且你的代码输出所有字符串都有‘\n’,fork之前,数据就全部被刷新,包括systemcall

2.重定向到log.txt,本质是向磁盘文件中写入(不是显示器!),我们系统对于数据的刷新方式有行刷新,变成了全缓冲!

3.全缓冲意味着缓冲区变大,实际写入的简单数据,不足以把缓冲区写满,fork执行的时候,数据依旧在缓冲区中。

4.我们目前的“缓冲区”,和操作系统是没有关系的 ,只能是语言本身有关,(用户缓冲区)

5.C/C++ 所提供的缓冲区,里面一定保存的是用户数据,属于当前进程在运行自己的数据

如果我们把数据交给操作系统,这个数据就属于操作系统,不属于自己(进程)。

6.当进程退出的时候一般要刷新缓冲区(修改了进程的数据),即使你的数据没有满足刷新条件。

总结:

因此在重定向后,由于变为全缓冲,C语言函数里要打印的东西就成为数据保存在C语言的缓冲区里,接着调用了fork函数创建了子进程,这时无论那个进程在退出前,都会刷新缓冲区(修改数据据,)向文件写入数据,接着就会发生写时拷贝,未退出的进程的数据中让然有之前缓冲区的数据,当这个进程退出是,还会刷新缓冲区,向文件写入数据。因此C语言函数的内容会打印两次。

write是系统调用函数,不使用C语言的缓冲区,直接写入操作系统,不属于进程,不发生写时拷贝!同时,这也是为什么write里的内容第一个打印到log.txt文件里。

4.什么是刷新?

这里我们用printf()函数举例。

在使用printf时,我们会把数据先弄到C语言的缓冲区,当出发刷新的条件时,就调用系统函数write,将缓冲区里的数据全部刷新到文件缓冲区里。而将C语言缓冲区里的数据弄到文件缓冲区里的的过程就是刷新。

为什么要采用刷新,而不是直接通过C语言函数,向文件缓冲区直接写入?

原因:采用刷新策略的效率比直接写入效率高。

文件缓冲区的数据如何刷新到文件上?

这个操作与我们无关,取决于操作系统,但是我们可以确定一点,文件缓冲区的刷新策略一定与语言缓冲区的刷新策略不同

C语言的缓冲区在哪里?

在任何情况下,我们输入输出的时候,都要有一个FILE,FILE是一个结构体,FILE里面除了包含fd(文件标识符)外,还有一段缓冲区。同时也是因此每打开一个文件,对应文件就会有个文件缓冲区,在我们对多个文件进行操作是,因为操作的缓冲区不同,是文件之间不会相互影响。

这里如果你想查看FILE结构体里有关缓冲区的内容,可以输入指令:

vim /usr/include/libio.h +246

结构体名是_IO_FILE与是C库对FILE使用typedef重命名了。

具体可通过指令:

vim /usr/include/stdio.h

打开文件后到第48行查看

仿写C库里的fopen,fclose,fwrite。

mystdio.h

#pragma once#define SIZE 4096        //缓冲区的大小//缓冲区刷新策略#define FLUSH_NONE 1         //直接刷新
#define FLUSH_LINE (1<<1)    //行刷新
#define FLUSH_ALL  (1<<2)    //全缓冲typedef struct _myFILE     //FILE结构体
{int fileno;         //文件标识符int flag;           //刷新策略char buffer[SIZE];  //语言缓冲区int end;            //缓冲区存储数据的大小
}myFILE;extern myFILE *my_fopen(const char *path, const char *mode);   //仿写fopenextern int my_fwrite(const char *s, int num, myFILE *stream);  //仿写fwriteextern int my_fflush(myFILE *stream);                          //仿写fflushextern int my_fclose(myFILE*stream);                           //仿写fclose



mystdio.c

#include "mystdio.h"
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <errno.h>
#include <stdlib.h>
#include <unistd.h>#define DFL_MODE 0666   //设置默认权限myFILE *my_fopen(const char *path, const char *mode)
{int fd   = 0;   int flag = 0;//设置文件打开方式if(strcmp(mode, "r") == 0){flag |= O_RDONLY;}else if(strcmp(mode, "w") == 0){flag |= (O_CREAT | O_TRUNC | O_WRONLY);}else if(strcmp(mode, "a") == 0){flag |= (O_CREAT | O_WRONLY | O_APPEND);}else{// Do Nothing}//文件不存在,创建文件并打开,并获取文件标识符与设置默认权限if(flag & O_CREAT){fd = open(path, flag, DFL_MODE);}//文件存在,打开文件,获取文件标识符else{fd = open(path, flag);}//判断文件是否打开成功if(fd < 0){errno = 2;return NULL;}//为FILE指针开辟空间myFILE *fp = (myFILE*)malloc(sizeof(myFILE));if(!fp) {errno = 3;return NULL;}//对FILE内容初始化fp->flag = FLUSH_LINE;fp->end = 0;fp->fileno = fd;return fp;
}int my_fwrite(const char *s, int num, myFILE *stream)  //仿写fwrite
{// 将数据写入C语言缓冲区memcpy(stream->buffer+stream->end, s, num);stream->end += num;// 判断是满足刷新条件, 这里我默认'\n'只会出现在结尾,在"abcd\nefgh"这种情况下会有bugif((stream->flag & FLUSH_LINE) && stream->end > 0 && stream->buffer[stream->end-1] == '\n'){my_fflush(stream);}return num;
}int my_fflush(myFILE *stream)//仿写fflush
{//当缓冲区不为空时,刷新缓冲区if(stream->end > 0){write(stream->fileno, stream->buffer, stream->end);//fsync(stream->fileno);stream->end = 0;}return 0;
}int my_fclose(myFILE*stream)   //仿写fclose
{my_fflush(stream);        //强制刷新return close(stream->fileno);
}

main.c(向文件中写入20次msg)

#include "mystdio.h"
#include <stdio.h>
#include <string.h>
#include <unistd.h>int main()
{myFILE *fp = my_fopen("./log.txt", "w");if(fp == NULL){perror("my_fopen");return 1;}int cnt = 20;const char *msg = "haha, this is my stdio lib";while(cnt--){my_fwrite(msg, strlen(msg), fp);sleep(1);}my_fclose(fp);return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/213073.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

b站pwn的学习总结

写的很乱 1.c语言的运行过程 了解了c语言需要经过以上2个过程&#xff08;编译和汇编&#xff09;&#xff0c;才能让机器按指令运行。机器只能听得懂机器码&#xff0c;所以要“汇编”。 那问题就来了&#xff0c;“编译”这个动作有啥用&#xff0c;c语言这种高级语言&…

玩转大数据10:深度学习与神经网络在大数据中的应用

目录 1. 引言&#xff1a;深度学习和神经网络在大数据中的重要性和应用场景 2. 深度学习的基本概念和架构 3. Java中的深度学习框架 3.1. Deeplearning4j框架介绍及Java编程模型 3.2. DL4J、Keras和TensorFlow的集成 4. 大数据与深度学习的结合 4.1. 大数据与深度学…

电脑端同时登录多个微信

1、建立一个txt文件 2、右击微信查看应用的属性&#xff0c;记录文件的位置 3、将步骤二得到的路径按照下方的格式输入到步骤一的文本中 4、保存之后将文本后缀名的.txt改成.bat 5、在未登录微信的情况下&#xff0c;双击即可得到两个微信登录窗口

解决idea 通过build project 手动触发热部署失败

在debug运行项目的过程中&#xff0c;并且保证&#xff08;不添加方法&#xff0c;不修改方法名&#xff09;一定的规则的情况下&#xff0c;可以通过build project 来手动热部署项目&#xff0c;也就是会交换class文件与resouces文件。 设置项 Edit Configurations Modify Op…

计算机图形学理论(1):建模基础

本系列根据国外一个图形小哥的讲解为本&#xff0c;整合互联网的一些资料&#xff0c;结合自己的一些理解。 场景的组成部分 场景相当于一个或多个模型的集合。模型包含以下内容&#xff1a; 结构描述&#xff1a;几何形状&#xff0c;如顶点、纹理坐标等表面描述&#xff1a…

Vue3中的defineModel

目录 一、vue3的defineModel介绍 二、defineModel使用 &#xff08;1&#xff09;在vite.config.js中开启 &#xff08;2&#xff09;子组件 &#xff08;3&#xff09;父组件 一、vue3的defineModel介绍 为什么要使用到defineModel呢&#xff1f;这里有这样一种场景&…

“快速排序:一种美丽的算法混沌”(1.hoare)

欢迎来到我的博客&#xff01;在今天的文章中&#xff0c;我将采用一种独特且直观的方式来探讨我们的主题&#xff1a;我会使用一幅图像来贯穿整篇文章的讲解。这幅精心设计的图表不仅是我们讨论的核心&#xff0c;也是一个视觉辅助工具&#xff0c;帮助你更深入地理解和掌握本…

学习深度强化学习---第2部分----RL动态规划相关算法

文章目录 2.1节 动态规划简介2.2节 值函数与贝尔曼方程2.3节 策略评估2.4节 策略改进2.5节 最优值函数与最优策略2.6节 值迭代与策略迭代2.7节 动态规划求解最优策略 本部分视频所在地址&#xff1a;深度强化学习的理论与实践 2.1节 动态规划简介 态规划有两种思路&#xff1…

前端 Web Workers 简介

简介 以前我们总说&#xff0c;JS 是单线程没有多线程&#xff0c;当 JS 在页面中运行长耗时同步任务的时候就会导致页面假死影响用户体验&#xff0c;从而需要设置把任务放在任务队列中&#xff1b;执行任务队列中的任务也并非多线程进行的&#xff0c;然而现在 HTML5 提供了…

App备案、ios备案Bundle ID查询、公钥信息、SHA-1值

App备案、ios备案Bundle ID查询、公钥信息、SHA-1值 Bundle ID这个就不说了&#xff0c;都知道是啥&#xff0c;主要说公钥信息和SHA-1值的获取 打开钥匙串访问&#xff0c;找到当前需要备案App的dis证书&#xff0c;如下&#xff1a; #####右键点击显示简介 #####可以看…

03.仿简道云公式函数实战-QLExpress初探

1. 前言 在上一篇文章中&#xff0c;我们简单介绍了一下表达式引擎&#xff0c;并引出我们的主角QLExpress.在这篇文章中&#xff0c;我们先来一个QLExpress的热身。 2. 初探QLExpress 源码地址&#xff1a;https://github.com/alibaba/qlExpress 笔者下载源码的版本是3.3.…

STL源码剖析笔记——适配器(adapters)

系列文章目录 STL源码剖析笔记——迭代器 STL源码剖析笔记——vector STL源码剖析笔记——list STL源码剖析笔记——deque、stack&#xff0c;queue STL源码剖析笔记——Binary Heap、priority_queue STL源码剖析笔记——AVL-tree、RB-tree、set、map、mutiset、mutimap STL源…

【Spring 基础】00 入门指南

【Spring 基础】00 入门指南 文章目录 【Spring 基础】00 入门指南1.简介2.概念1&#xff09;控制反转&#xff08;IoC&#xff09;2&#xff09;依赖注入&#xff08;DI&#xff09; 3.核心模块1&#xff09;Spring Core2&#xff09;Spring AOP3&#xff09;Spring MVC4&…

php实现截取姓名中的第一个字作为头像的实战记录

php 截取中文字符串第一个字 substr 函数 在 PHP 中&#xff0c;使用 substr 函数来截取中文字符串的第一个字。由于 PHP 默认的字符编码是 UTF-8&#xff0c;它可以正确处理中文字符。 $chineseString "你好世界"; $firstChar substr($chineseString, 0, 1); e…

中文分词演进(查词典,hmm标注,无监督统计)新词发现

查词典和字标注 目前中文分词主要有两种思路&#xff1a;查词典和字标注。 首先&#xff0c;查词典的方法有&#xff1a;机械的最大匹配法、最少词数法&#xff0c;以及基于有向无环图的最大概率组合&#xff0c;还有基于语言模型的最大概率组合&#xff0c;等等。 查词典的方法…

知识产权服务企业网站建设效果如何

知识产权服务也有较高的市场需求度&#xff0c;尤其如今互联网深入到各个行业&#xff0c;无论个人还是企业都会以不同的方式经营&#xff0c;相应的为保障自身权益&#xff0c;注册商标、专利等自然不可少&#xff0c;而对普通小白来说&#xff0c;想要完成这些流程也是有些难…

Python实现获取b站视频的弹幕内容

前言 本文是该专栏的第39篇,后面会持续分享python的各种干货知识,值得关注。 在本专栏之前,有详细介绍使用python增加b站视频的播放量方法,感兴趣的同学可往前翻阅《Python-增加b站视频播放量》。而本文,笔者再来单独的详细介绍,通过python来获取b站视频的弹幕内容。如下…

CGAL的3D皮肤表面网格

1、介绍 Edelsbrunner 引入的皮肤表面和具有丰富而简单的组合和几何结构&#xff0c;使其适合在生物计算中模拟大分子。 对这些表面进行网格划分通常是进一步处理其几何形状所必需的&#xff0c;例如在数值模拟和可视化中。 皮肤表面由一组加权点&#xff08;输入球&#xff09…

html通过CDN引入Vue使用Vuex以及Computed、Watch监听

html通过CDN引入Vue使用Vuex以及Computed、Watch监听 近期遇到个需求&#xff0c;就是需要在.net MVC的项目中&#xff0c;对已有的项目的首页进行优化&#xff0c;也就是写原生html和js。但是咱是一个写前端的&#xff0c;写html还可以&#xff0c;.net的话&#xff0c;开发也…

期末速成数据库极简版【查询】(3)

目录 多表查询 【8】多表连接——内连接 &#x1f642;等值连接 &#x1f642;自然连接 &#x1f642;非等值连接 【9】多表连接——外连接 【10】交叉连接不考 【11】联合查询 【12】扩展多表连接 【13】嵌套查询 &#x1f642; 多表查询 【8】多表连接——内连…