【Linux系统化学习】缓冲区

2024-02-17 23:28

本文主要是介绍【Linux系统化学习】缓冲区,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

缓冲区

 一个样例

现象解释

缓冲区存在的位置


缓冲区

在刚开始学习C语言的时候我们就听过缓冲区这个名词,很是晦涩难懂;在Linux下进程退出时也包含缓冲区,因此缓冲区到底是什么?有什么作用?

让我们先从一个小故事说起:

身在西藏上大学的张三有一个在海南的朋友李四,李四马上过生日了张三想要送李四一个键盘;于是张三定了各种交通工具的票,自己带着键盘千里迢迢的跑到海南张三的楼下送给张三,又千里迢迢的原路返回,这一来回耽误了张三好多的时间和精力。

又是一年,李四又过生日了;张三想送李四一个鼠标。打听到自己家和张三家楼下都有菜鸟驿站于是将鼠标交给菜鸟驿站,通过菜鸟驿站的快递小哥千里迢迢的将鼠标送到李四家楼下的菜鸟驿站。张三自己不用跑这一个来回。但是菜鸟驿站不可能因为张三一个鼠标进行派送,先暂存一部分快递要等到快递车装满进行派送,或者一行存储柜子满了在派送。但是也有可能特殊情况顾客“加钱”,想要直接派送。

上面这个故事中的菜鸟驿站就是我们口中的缓冲区

缓冲区的作用提高使用者的效率

因此缓冲区是内存空间的一部分。也就是说,在内存空间中预留了一定的存储空间,这些存储空间用来缓冲输入或输出的数据,这部分预留的空间就叫做缓冲区。

暂存快递等到快递车装满派送:全缓冲(缓冲区满了刷新)

“加钱”直接派送:无缓冲(立即刷新)

暂存一行存储柜派送:行缓冲(一行满了刷新)

这是一般的刷新策略,特殊情况

  • 强制刷新
  • 进程推出的时候,一般要进行刷新缓冲区

一般对于显示器文件:行刷新(行缓冲),对于磁盘上的文件:全缓冲

 一个样例

 1 #include<stdio.h>2 #include<string.h>3 #include<unistd.h>                                                                                                                                       4 int main()5 {6     fprintf(stdout,"hello fprintf\n");7     printf("hello print\n");8     fputs("hello fputs\n",stdout);9 10     //系统调用11     const char * str = "system call : hello write\n";12     write(1,str,strlen(str));13     return 0;14 }

  1 #include<stdio.h>2 #include<string.h>3 #include<unistd.h>4 int main()5 {6     fprintf(stdout,"hello fprintf\n");7     printf("hello print\n");8     fputs("hello fputs\n",stdout);9 10     //系统调用11     const char * str = "system call : hello write\n";12     write(1,str,strlen(str));13     fork();                                                                                                                                              14     return 0;15 }

上面第一种情况在最后没有创建子进程直接执行和重定向到某个文件中也没有什么问题,就是在重定向时系统调用接口先执行;

第二种情况是在程序运行最后使用fork函数创建子进程直接执行程序时没有问题,但是在重定向时除了系统调用,剩下的语句被执行了两次。

现象解释

  • 当我们直接向显示器进行打印的时候,显示器文件的刷新方式是行刷新。而代码输出的所有字符串,都有”\n",fork之前数据已经全部刷新包括system call。
  • 重定向到log.txt,本质是向磁盘文件中写入,我们系统对于数据的刷新方式已经由行刷新变成了全缓冲。
  • 全缓冲意味着缓冲区变大,实际写入的简单数据不足以把缓冲区写满,fork执行的时候数据依旧在缓冲区中。
  • 当前阶段的缓冲区是用户缓冲区是C语言和操作系统没有任何关系。
  • C/C++提供的缓冲区,里面保存的是用户的数据,是属于当前进程;如果把这个数据交给操作系统,这个数据就数据操作系统。
  • 当进程推出的时候,一般要进行刷新缓冲区,即使这个数据没有满足刷新条件;刷新缓冲区属于对文件的写入操作;fork立马退出,任意一个进程在推出的时候都会刷新缓冲区,就要发生写时拷贝。
  • 系统调用没有使用C语言的缓冲区,直接写入到操作系统,不属于进城了不发生写时拷贝。
  • 从C语言的缓冲区写入到操作系统中这个过程就是刷新

缓冲区存在的位置

 

我们在使用C语言的一些文件操作接口时会发现很多函数的的返回值或者函数参数的类型为FILE*,之前的文章中我们提到过Linux下的文件描述符也是在FILE中的;不难猜测出C语言中的缓冲区存在在FILE中。当我们查看C语言的源码可以证实这一点:

我们来看看FILE的部分结构

typedef struct _IO_FILE FILE; 在/usr/include/stdio.h

在/usr/include/libio.h
struct _IO_FILE {
int _flags; /* High-order word is _IO_MAGIC; rest is flags. */
#define _IO_file_flags _flags
//缓冲区相关
/* The following pointers correspond to the C++ streambuf protocol. */
/* Note: Tk uses the _IO_read_ptr and _IO_read_end fields directly. */
char* _IO_read_ptr; /* Current read pointer */
char* _IO_read_end; /* End of get area. */
char* _IO_read_base; /* Start of putback+get area. */
char* _IO_write_base; /* Start of put area. */
char* _IO_write_ptr; /* Current put pointer. */
char* _IO_write_end; /* End of put area. */
char* _IO_buf_base; /* Start of reserve area. */
char* _IO_buf_end; /* End of reserve area. */
/* The following fields are used to support backing up and undo. */
char *_IO_save_base; /* Pointer to start of non-current get area. */
char *_IO_backup_base; /* Pointer to first valid character of backup area */
char *_IO_save_end; /* Pointer to end of non-current get area. */
struct _IO_marker *_markers;
struct _IO_FILE *_chain;
int _fileno; //封装的文件描述符
#if 0
int _blksize;
#else
int _flags2;
#endif
_IO_off_t _old_offset; /* This used to be _offset but it's too small. */
#define __HAVE_COLUMN /* temporary */
/* 1+column number of pbase(); 0 is unknown. */
unsigned short _cur_column;
signed char _vtable_offset;
char _shortbuf[1];
/* char* _save_gptr; char* _save_egptr; */
_IO_lock_t *_lock;
#ifdef _IO_USE_OLD_IO_FILE
};

今天对Linux下缓冲区的分享到这就结束了,希望大家读完后有很大的收获,也可以在评论区点评文章中的内容和分享自己的看法;个人主页还有很多精彩的内容。您三连的支持就是我前进的动力,感谢大家的支持!!! 

 

这篇关于【Linux系统化学习】缓冲区的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/719314

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

linux-基础知识3

打包和压缩 zip 安装zip软件包 yum -y install zip unzip 压缩打包命令: zip -q -r -d -u 压缩包文件名 目录和文件名列表 -q:不显示命令执行过程-r:递归处理,打包各级子目录和文件-u:把文件增加/替换到压缩包中-d:从压缩包中删除指定的文件 解压:unzip 压缩包名 打包文件 把压缩包从服务器下载到本地 把压缩包上传到服务器(zip

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

Linux 网络编程 --- 应用层

一、自定义协议和序列化反序列化 代码: 序列化反序列化实现网络版本计算器 二、HTTP协议 1、谈两个简单的预备知识 https://www.baidu.com/ --- 域名 --- 域名解析 --- IP地址 http的端口号为80端口,https的端口号为443 url为统一资源定位符。CSDNhttps://mp.csdn.net/mp_blog/creation/editor

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss