最新资讯

  • linux基础IO——用户缓冲区——概念深度探索、IO模拟实现

linux基础IO——用户缓冲区——概念深度探索、IO模拟实现

2025-05-02 22:37:46 0 阅读

        前言:本篇文章主要讲解文件缓冲区。 讲解的方式是通过抛出问题, 然后通过分析问题, 将缓冲区的概念与原理一步一步地讲解。同时, 本节内容在最后一部分还会带友友们模拟实现一下c语言的printf, fprintf接口, 加深友友们对于缓冲区的理解。

        ps: 本节内容适合了解linux进程和linux文件重定向的友友们进行观看。  

目录

缓冲区概念——使用close引出

exit和用户缓冲区

缓冲区刷新方案

为什么要有用户缓冲区

FILE

fork和缓冲区读写

模拟实现c语言标准库

只包含fd的接口——不带缓冲区

c语言的跨平台性

带缓冲区的c语言接口

fflush


缓冲区概念——使用close引出

        讲述缓冲区, 我们需要先看一下下面的接口:

        对于上面的接口, 我们可以预测一下——就是打印4行内容, 分别是hello printfhello fprintfhello fwritehello write

        如图,代码运行结果符合我们的预期。对于上面的代码我们知道, 对于c语言的接口, stdoutFILE文件流里面就封装了1号文件描述符。 他们的底层一定是调用了write函数。

    对于上面的函数,我们重定向到文件是这样, 和直接在显示器上面输出是一样的。

  • 但是, 如果加上close(1), 将显示器文件关闭, 那么情况就有些不同了, 下面是有 的:

  • 下面是没有 的:

        如果没有了 , 对于c语言的打印接口, 就不能输出了。

        如下图, 只输出了系统调用write:

        我们知道对于上面printf、fprintf、fwrite来说, 这三个是c语言接口, 他们的底层一定是封装的write接口。 这就有问题了, 为什么printf、fprintf、fwrite的底层是write, 而write能够打印数据, printf、fprintf、fwrite都不能打印数据呢?

        友友们如果学习过进度条, 那么就会知道一个缓冲区的概念。 这个缓冲区遇到 或者缓冲区满了之后就会刷新自己。 

        一般情况下, 我们使用printf、fprintf、fwrite不是直接写向显示器, 而是先写向缓冲区。 等到遇到 或者缓冲区满了之后再刷新缓冲区,将数据刷新到缓冲区。

        也就是说, 对于上面的这个代码来说, 当程序运行到close(1)前一行, 那么数据就已经写到缓冲区了。 只是这个缓冲区一定不在操作系统内部, 这个缓冲区不是系统级别的缓冲区!!!

        具体的底层逻辑关系如下:

        我们使用的c语言接口的时候, 如果是将数据直接拷贝到了系统级别的缓冲区中, 那么等到close(1)的时候, 这个close(1)就能找到对应的文件struct file, 然后找到对应的系统级别的文件缓冲区, 将缓冲区内容刷新到磁盘里面。 也就是说, 如果c语言接口是将数据只拷贝到系统级别的缓冲区, 那么我们应该也能看到c语言接口的结果, 而事实上并没有。 而write是系统调用接口, 是直接写到系统级别的文件缓冲区, 那么当close(1)的时候, 就能直接将数据刷新到磁盘里面。

        我们平时口中的缓冲区, 不是这个系统级别的缓冲区, 而是语言提供的, 这里也就是c语言提供的一个缓冲区, 这个缓冲区是一个用户级别的缓冲区。 也就是说, c语言的printf, fprintf等接口是将数据写到语言级别的缓冲区中, 然后等到某一个时机就调用write接口, 将数据刷新到系统级别的缓冲区中,最后再刷新到磁盘中

        

        所以, 当我们printf、fprintf等等的时候, 这些数据其实是保存在语言级别的缓冲区, 他们还没有进入系统级别的缓冲区, 当close(1)的时候。1号文件struct file, 系统文件缓冲区都被关掉了。 就无法再向磁盘写入数据了, 也就没有显示结果。

        显示器对应的刷新方案是行刷新, 当我们在c语言接口要打印的数据中加上 的时候, 如下图, 那么即便有close(1), 但是对于用户缓冲区来说, 每一行都会被刷新到系统级别的文件缓冲区。 所以可以打印出来。 用户缓冲区刷新的本质就是将数据通过1 + write写到内核中。

exit和用户缓冲区

        我们之前说过exit和_exit, exit是封装的上层接口, 而_exit是系统调用。 两者的区别是_exit退出前不会刷新缓冲区(我们口中的缓冲区, 都是用户级缓冲区), 而exit会刷新缓冲区(我们口中的缓冲区, 都是用户级缓冲区)。 这是因为_exit身为系统调用, 它在用户层的下层, 它根本看不到用户级缓冲区。 也就没办法刷新它, 而exit处在用户层, 它底层一定是先刷新缓冲区flush, 然后再调用系统调用_exit。

        从上面我的论述中就可以发现, 其实当数据到达内核的时候, 那么这个数据就可以到达硬件了。 ——这也是我们目前认为的, 只要是将数据刷新到了内核, 数据就可以到达硬件了!!!

缓冲区刷新方案

        缓冲区刷新问题:——这里的缓冲区考虑的是语言层, 也就是用户层缓冲区。

  •         无缓冲: 直接刷新, 不管printf, fprintf等接口打印的数据是什么, 直接刷新。
  •         行缓冲: 不刷新,直到遇到 再刷新。
  •         全缓冲: 缓冲区满了再刷新, 不满的话, 无论如何都不刷新。

 所以整个的刷新流程如下图:

        在向显示器中写入的时候, 采用的是行缓冲。

        在向普通文件中写入的时候, 采用的是全缓冲。

为什么要有用户缓冲区

        进程退出的时候也会刷新缓冲区, 为什么要有这个缓冲区呢? 

        首先就是效率层面——这里可以使用一个例子来说明这件事, 就比如我们给朋友送东西, 如果没有快递公司, 我们就必须亲自去送东西给朋友, 但是现在有了快递公司, 快递公司就可以帮助我们去送快递, 我们节省的这些时间就可以去做自己的事情。——这里面的快递公司就是缓冲区, 我们就是用户。 所以, 缓冲区解决的是谁的效率问题?——其实就是用户的效率问题。 (也就是使用c语言的人的效率问题)

        其次是语言设计层面——拥有缓冲区的目的是为了配合c语言接口:printf,fprintf的输出格式化。——这里我们需要想一个问题, 就是我们在显示器上面打印的789, 这样的数字, 打印的是字符呢? 打印的还是数字呢? ——答案是打印的字符789。 也就是说, 将789转化成字符7, 字符8, 字符9, 然后再打印到显示器上面。 当数据打印的时候, 先将数据打印到用户缓冲区。 当某个时机的时候, 就将用户缓冲区的数据打印到硬件里面。 由于这种数据进入, 数据流出, 很像一条河流, 所以用户缓冲区也被叫做流。
 

FILE

请问, FILE是属于用户呢? 还是属于操作系统呢? 答案是用户, 语言都是属于用户的。

        知道这个之后, 我们就可以知道, 对于fopen来说

        如果使用fopen打开了一个文件, 那么就会获取这个文件的文件描述符fd。 然后再malloc一块内存空间保存在FILE, 这个fd和malloc的内存空间都保存在FILE里面!!也就是下面这张图:

fork和缓冲区读写

现在,我这里有另外一个关于fork函数的问题。 我们看下面这串代码:

        我们生成程序后, 将数据重定向到文件中。 如果按照我们以往的经验的话, 这里我们预测会打印四行内容, 分别是hello printf、hello fprintf、hello fwrite、hello write

        但是, 我们实际运行的结果如下:

        如上图, 我们可以发现c语言接口的打印都被打印了两次, 只有write系统调用被打印了一次。

        这是因为当使用重定向, 重定向到了文件之后, 缓冲区的刷新方案就变成了全缓冲, 遇到 不再刷新, 而是等缓冲区被写满之后才刷新。

        那么我们下载来证明一下重定向到文件是全缓冲的

        首先我们写下面这个程序;

        这个程序如果按照我们的猜想。 前三秒不会打印任何数据, 这些数据都会被打印到了缓冲区之中。 当三秒过后, write会被打印。然后再五秒所有数据才会全部被打印。

         我们先打开监视窗口进行观察:

然后运行这个程序, 就会得到:

        那么这是为什么呢?——首先我们知道, 当我们三个c语言接口执行完的时候, 数据都被写到了缓冲区中。 而缓冲区也是在进程里面的, 所以当fork创建子进程后, 再刷新缓冲区, 而刷新缓冲区就相当于数据的修改, 需要进行写时拷贝。 所以就有两份缓冲区, 两个缓冲区的数据都被刷新后就产生了上面的现象!!!

模拟实现c语言标准库

        为了模拟实现c语言的标准库, 我们创建三个文件。 这里的_my_func.h就相当于stdio.h这样的标准库, 而_my_func.c就是用来模拟实现c标准库接口。

         我们打开_my_func.h, 首先做好预备工作。 先让该头文件不可重复包含, 也就是下面的ifdef, define, endif。 然后再包含一个头文件string.h。 我们先来简单的实现以下三个函数, _fopen, _fwrite, _fclose

只包含fd的接口——不带缓冲区

下面是代码:

#include

//定义缓冲区最大长度
#define SIZE 1024

//flag用来表示刷新方案
#define FLUSH_NOw 0
#define FLUSH_LINE 1
#define FLUSH_ALL 2

typedef struct IO_FILE
{
  int fileno; //文件fd                                                       
  int flag;

  //输入缓冲区
  //char inbuffer[SIZE];
  //int in_pos;
  //输出缓冲区
  char outbuffer[SIZE];    
  int out_pos;    
}_FILE;

_FILE* _fopen(const char* filename, const char* mode);
int _fwrite(_FILE* fp, const char* s, int len);
void _fclose(_FILE* fp);
void _fflush(_FILE* fp);

#endif

然后我们定义FILE类型的结构体。 但是为了区分标准库。 我们这里改成_FILE(以后的函数等等为了区分都会加上"_")。

        对于这个结构体里面的内容, 根据我们前面讲到的只是可以知道, 这里面一定有两个字段一个是文件描述符fd, 一个是文件缓冲区。 这里我们先将fd包含进来。如下图:

然后我们就可以先简单的实现一下fopen, fwrite, fclose函数了。

fopen

如下为代码:

_FILE* _fopen(const char* filename, const char* mode)
  {
    //先打开文件                       
    //open函数打开文件, 返回文件描述符。第二个参数是打开方式
    //先判断打开文件的方式。确定第二个参数                                   
    int f = 0;                
    int fd = -1; //一开始令fd 为-1
                                              
    if (strcmp(mode, "w") == 0) f = (O_CREAT|O_WRONLY|O_TRUNC); 
    else if (strcmp(mode, "a") == 0) f = (O_CREAT|O_WRONLY|O_APPEND);
    else if (strcmp(mode, "r") == 0) f = (O_RDONLY);
    else return NULL;
   
    //打开文件, 默认权限是FILEMODE
    fd = open(filename, f, FILEMODE);
    //如果fd是-1, 那么直接return。 否则就是正常打开文件。 正常进行操作
    if (fd == -1) return NULL;
    //else                          
    _FILE* fp = (_FILE*)malloc(sizeof(_FILE));
    if (fp == NULL) return NULL;//如果fp == NULL说明空间不够了。
    //else
    fp->fileno = fd;//让将fd赋值给fp的fd。
    return fp;
  }

fwrite

 如下为代码:

 int _fwrite(_FILE* fp, const char* s, int len)                             
  {
    //fd, 要写入的字符串, 字节长度
    return write(fp->fileno, s, len);
  }

 fclose

如下为代码:

  //关闭文件                     
  void _fclose(_FILE* fp)        
  {                              
    if (fp == NULL) return;      
    close(fp->fileno);           
    free(fp);                    
  }   

 我们自己实现一串代码进行测试, 打开的方式是清空写:

运行结果如下:

         其实从上面的代码我们就能看到封装的好处。 有了fopen, fwrite这些封装之后, 以后再向文件中写内容, 我们想要修改写入的方式, 就不需要再使用O_RONLY、O_WRONLY这些标志了, 直接使用w, a, r, 然后程序就会自动帮助我们判断如何打开文件!!!

c语言的跨平台性

        我们在上面实现的接口是符合linux环境的接口。 如果我们再windows下, 想实现同样的接口, 那么为了能够在windows正常工作,就要实现一份适合在windows下面跑的代码。 同样的macos也是一样的。 这三份代码我们可以使用if else if, endif条件编译。 如果在linux下面就是用linux下面的代码。 如果在windows下面就是用windows下面的代码。 如果在macos下面, 就是用macos下的代码。 这个就叫做c语言的跨平台性!!!

带缓冲区的c语言接口

我们在我们的接口里面加入缓冲区, 下图是引入缓冲区的方法。图中的in_pos和out_pos是为了标记缓冲区的使用情况。 它指向缓冲区已经使用的最后一个位置。 这个位置的左边是已经使用的, 右边是没有使用的。而且由于我们此次模拟实现的接口用不到输入缓冲区, 所以将输入缓冲区注释掉。

接下来我们修改我们的代码:

fopen 

      首先修改fopen, 要初始化缓冲区, 那么就是将标记缓冲区使用情况的out_pos置为0. 就代表缓冲区没有被使用过。 并且我们要初始化缓冲区的刷新方案。也就是_FILE结构体里面的flag。 我们这里默认初始化为行刷新, 下图的黄色代码就是初始化缓冲区。(只需要在原本的fopen下面添加黄色框框的代码就可以实现)

fwrite

        如下图是写方法的含有缓冲区的实现方式。 就是先将数据拷贝到缓冲区中, 然后判断刷新方式, 是直接刷新, 行刷新还是缓冲区满才刷新。刷新后标记位out_pos置为0.

下面是代码:


  //写方法
  int _fwrite(_FILE* fp, const char* s, int len)
  {
    //使用memcpy将数据拷贝到输出缓冲区中。 
    memcpy(&fp->outbuffer[fp->out_pos], s, len);
    fp->out_pos += len;
    //判断刷新方式
    //直接刷新
    if (fp->flag == FLUSH_NOw)
    {
      write(fp->fileno, fp->outbuffer, fp->out_pos);
      fp->out_pos = 0;
    }                                                                        
    //遇到反斜杠n刷新
    else if ((fp->flag == FLUSH_LINE) && (fp->outbuffer[fp->out_pos - 1] == '  
'))
    {
      //只要遇到反斜杠n就行刷新
      write(fp->fileno, fp->outbuffer, fp->out_pos);
      fp->out_pos = 0;
    }
    else if (fp->flag == FLUSH_ALL && fp->out_pos == SIZE) 
    {
      write(fp->fileno, fp->outbuffer, fp->out_pos);
      fp->out_pos = 0;
    }
    else return len;
    return len;
  }

fclose

关闭文件和之前没有什么区别

测试:

         我们可以使用下面这个监控脚本, 以及我们要测试的代码, 来测试一下我们模拟实现的是否正确:

#include"_my_func.h"    
#include    
    
int main()    
{    
  _FILE* fp = _fopen("mytest.txt", "a");    
  if (fp == NULL) return 1;    
    
  int cnt = 10;    
  const char* mes = "hello linux
";    
  while (cnt)    
  {    
    _fwrite(fp, mes, strlen(mes));    
    sleep(1);    
    cnt--;    
  }    
  _fflush(fp);                                                               
  _fclose(fp);    
  return 0;    
}    

运行结果如下, 因为我们是以行刷新, 并且每一秒都会追加一行。 所以会出现下图的情况。 验证结果我们的代码是正确的。

 

fflush

        fflush这里我们也要模拟一下, 博主模拟是为了应对全刷新的情况(并不是说fflush只是为了应对全刷新而存在的)。因为我们使用全刷新的时候, 写到缓冲区的内容不容易被刷新出来。就如同下图我们已经运行了程序, 但是仍旧刷新不出东西, 也就是没有向文件写入:

        那么实现fflush, 我们要怎么实现呢?代码如下图:

代码: 


  void _fflush(_FILE* fp)
  {
    if (fp->out_pos > 0) 
    {                                                                        
      write(fp->fileno, fp->outbuffer, fp->out_pos);
      fp->out_pos = 0;
    }
  }
  
  
  //关闭文件 
E>void _fclose(_FILE* fp)
  {
    if (fp == NULL) return;
    _fflush(fp);//因为关闭文件要将缓冲区中的东西都放出来。
    close(fp->fileno);
    free(fp);
  }

         我们在运行就能在10秒胡进程退出的时候打出来了。下图就是测试——前十秒没有打印任何东西, 但是后面打印了一串数据。 这是因为进程退出的时候刷新了缓冲区。

以上就是本节的全部内容。 下面是博主的笔记:

 

本文地址:https://www.vps345.com/7350.html

搜索文章

Tags

PV计算 带宽计算 流量带宽 服务器带宽 上行带宽 上行速率 什么是上行带宽? CC攻击 攻击怎么办 流量攻击 DDOS攻击 服务器被攻击怎么办 源IP 服务器 linux 运维 游戏 云计算 javascript 前端 chrome edge llama 算法 opencv 自然语言处理 神经网络 语言模型 python MCP harmonyos 华为 开发语言 typescript 计算机网络 ssh ubuntu 数据库 centos oracle 关系型 安全 分布式 阿里云 网络 网络安全 网络协议 deepseek Ollama 模型联网 API CherryStudio RTSP xop RTP RTSPServer 推流 视频 科技 ai java 人工智能 个人开发 进程 操作系统 进程控制 Ubuntu vue.js audio vue音乐播放器 vue播放音频文件 Audio音频播放器自定义样式 播放暂停进度条音量调节快进快退 自定义audio覆盖默认样式 macos adb C# MQTTS 双向认证 emqx ssl 开源 docker github 实时音视频 实时互动 EtherCAT转Modbus ECT转Modbus协议 EtherCAT转485网关 ECT转Modbus串口网关 EtherCAT转485协议 ECT转Modbus网关 fastapi mcp mcp-proxy mcp-inspector fastapi-mcp agent sse 物联网 tcp/ip mcu iot 信息与通信 filezilla 无法连接服务器 连接被服务器拒绝 vsftpd 331/530 智能路由器 外网访问 内网穿透 端口映射 numpy 嵌入式 linux驱动开发 arm开发 嵌入式硬件 rust http HarmonyOS Next 运维开发 云原生 pip conda ui 华为云 华为od Flask FastAPI Waitress Gunicorn uWSGI Uvicorn debian PVE android 鸿蒙 YOLO efficientVIT YOLOv8替换主干网络 TOLOv8 asm 面试 性能优化 jdk intellij-idea 架构 宝塔面板访问不了 宝塔面板网站访问不了 宝塔面板怎么配置网站能访问 宝塔面板配置ip访问 宝塔面板配置域名访问教程 宝塔面板配置教程 pycharm ide pytorch Windsurf 电脑 .net vim mysql word图片自动上传 word一键转存 复制word图片 复制word图文 复制word公式 粘贴word图文 粘贴word公式 VMware安装mocOS VMware macOS系统安装 容器 windows flask spring boot AI编程 AIGC json java-ee jar gradle elasticsearch uni-app websocket 产品经理 agi microsoft ESP32 camera Arduino 电子信息 大数据 政务 分布式系统 监控运维 Prometheus Grafana 后端 豆瓣 追剧助手 迅雷 nas 微信 tcpdump 银河麒麟 kylin v10 麒麟 v10 深度学习 目标检测 计算机视觉 c++ 单片机 AI Agent 编辑器 鸿蒙系统 前端框架 aws googlecloud LDAP rpc HCIE 数通 计算机外设 mac 软件需求 笔记 tomcat maven intellij idea vscode docker搭建nacos详解 docker部署nacos docker安装nacos 腾讯云搭建nacos centos7搭建nacos nginx dubbo gateway Clion Nova ResharperC++引擎 Centos7 远程开发 pillow node.js html5 firefox YOLOv12 rust腐蚀 DevEco Studio 微信分享 Image wxopensdk WSL win11 无法解析服务器的名称或地址 kubernetes sqlserver 机器学习 向日葵 学习方法 经验分享 程序人生 vSphere vCenter 软件定义数据中心 sddc golang 迁移指南 ffmpeg 音视频 数据结构 学习 c语言 温湿度数据上传到服务器 Arduino HTTP rag ragflow ragflow 源码启动 DeepSeek-R1 API接口 cuda cudnn anaconda web安全 中间件 iis 微服务 springcloud ESXi 命名管道 客户端与服务端通信 udp 客户端 live555 rtsp rtp Linux 进程信号 Hyper-V WinRM TrustedHosts Qwen2.5-coder 离线部署 ollama 大模型 unix vue3 HTML audio 控件组件 vue3 audio音乐播放器 Audio标签自定义样式默认 vue3播放音频文件音效音乐 自定义audio播放器样式 播放暂停调整声音大小下载文件 mount挂载磁盘 wrong fs type LVM挂载磁盘 Centos7.9 Dify k8s 自动化 C 环境变量 进程地址空间 源码剖析 rtsp实现步骤 流媒体开发 log4j apache c# .netcore 创意 社区 ue5 vr cpu 内存 实时 使用 僵尸进程 ipython flutter react.js 前端面试题 持续部署 YOLOv8 NPU Atlas800 A300I pro asi_bench jenkins 统信UOS 麒麟 bonding 链路聚合 ddos qt stm32项目 stm32 jmeter 软件测试 爬虫 数据集 gpu算力 zotero WebDAV 同步失败 代理模式 gitee 并查集 leetcode unity dell服务器 go WSL2 GaN HEMT 氮化镓 单粒子烧毁 辐射损伤 辐照效应 git sql KingBase ai小智 语音助手 ai小智配网 ai小智教程 智能硬件 esp32语音助手 diy语音助手 protobuf 序列化和反序列化 安装 oceanbase rc.local 开机自启 systemd 智能手机 NAS Termux Samba chatgpt llama3 Chatglm 开源大模型 php 安装教程 GPU环境配置 Ubuntu22 CUDA PyTorch Anaconda安装 Dell R750XS postman mock mock server 模拟服务器 mock服务器 Postman内置变量 Postman随机数据 MacOS录屏软件 低代码 数据挖掘 网络用户购物行为分析可视化平台 大数据毕业设计 1024程序员节 mamba nuxt3 华为认证 网络工程师 拓扑图 产测工具框架 IMX6ULL 管理框架 远程桌面 AI大模型 LLM CH340 串口驱动 CH341 uart 485 ruoyi visual studio code cmos 硬件 IIS服务器 IIS性能 日志监控 react next.js 部署 部署next.js 腾讯云 ocr 银河麒麟服务器操作系统 系统激活 重启 排查 系统重启 日志 原因 r语言 数据可视化 数据分析 P2P HDLC 思科 医疗APP开发 app开发 远程工作 算力 ansible linux上传下载 llm 无人机 串口服务器 程序员 TRAE 其他 RAID RAID技术 磁盘 存储 windwos防火墙 defender防火墙 win防火墙白名单 防火墙白名单效果 防火墙只允许指定应用上网 防火墙允许指定上网其它禁止 dify 高效远程协作 TrustViewer体验 跨设备操作便利 智能远程控制 机器人 spring cloud kafka hibernate excel linux安装配置 KylinV10 麒麟操作系统 虚拟机 Vmware 漏洞 bash kali 共享文件夹 docker compose kind 系统开发 binder 车载系统 framework 源码环境 负载均衡 postgresql 宝塔面板 同步 备份 建站 安全威胁分析 Trae AI代码编辑器 grafana prometheus ci/cd IPMI redis 缓存 bug 弹性计算 云服务器 裸金属服务器 弹性裸金属服务器 虚拟化 express p2p gpt-3 文心一言 系统架构 设计模式 软件工程 iBMC UltraISO openEuler Linux PID unity3d 网络穿透 可信计算技术 安全架构 网络攻击模型 rocketmq SSH Xterminal fpga开发 DeepSeek 大模型入门 腾讯云大模型知识引擎 Deepseek 灵办AI kvm 序列化反序列化 C++ wsl 致远OA OA服务器 服务器磁盘扩容 etcd 数据安全 RBAC 企业微信 Linux24.04 deepin gitlab okhttp CORS 跨域 树莓派 VNC 金融 雨云 NPS 单元测试 集成测试 功能测试 selenium dns CentOS Stream CentOS rabbitmq webrtc 报错 Java 换源 镜像源 国内源 Debian 测试工具 kylin arm 能力提升 面试宝典 技术 IT信息化 大模型微调 线程 多线程 code-server MQTT mosquitto 消息队列 媒体 JAVA 开发 django sqlite 蓝桥杯 交换机 MS Materials wsl2 openssl 密码学 监控 自动化运维 https springboot远程调试 java项目远程debug docker远程debug java项目远程调试 springboot远程 Reactor trae 大数据平台 lio-sam SLAM 多层架构 解耦 echarts 信息可视化 网页设计 uv 微信小程序 AP配网 AK配网 小程序AP配网和AK配网教程 WIFI设备配网小程序UDP开 硬件架构 AISphereButler 课程设计 shell kamailio sip VoIP mariadb 国产操作系统 ukui 麒麟kylinos openeuler 统信 虚拟机安装 多进程 框架搭建 Node-Red 编程工具 流编程 spring wps 安卓 Java Applet URL操作 服务器建立 Socket编程 网络文件读取 Redis Desktop mybatis 大模型教程 remote-ssh 分析解读 ecmascript KVM C语言 小程序 pdf n8n 工作流 workflow RustDesk自建服务器 rustdesk服务器 docker rustdesk firewalld web3.py 办公自动化 自动化生成 pdf教程 VPS pyqt rclone AList webdav fnOS 孤岛惊魂4 WebRTC gpt 小艺 Pura X 恒源云 arcgis RTMP 应用层 jupyter wireshark 系统安全 飞书 matplotlib Cookie ue4 着色器 虚幻 pyautogui 目标跟踪 OpenVINO 推理应用 asp.net大文件上传 asp.net大文件上传源码 ASP.NET断点续传 asp.net上传文件夹 asp.net上传大文件 .net core断点续传 .net mvc断点续传 mongodb adobe 传统数据库升级 银行 大语言模型 LLMs 开机自启动 ShenTong 国产化 Python 网络编程 聊天服务器 套接字 TCP Socket IPMITOOL BMC 硬件管理 opcua opcda KEPServer安装 oneapi Portainer搭建 Portainer使用 Portainer使用详解 Portainer详解 Portainer portainer virtualenv safari Mac 系统 服务器管理 配置教程 服务器安装 网站管理 nvidia playbook 剧本 Vmamba QQ bot Docker 多线程服务器 Linux网络编程 云服务 springsecurity6 oauth2 授权服务器 token sas FTP 服务器 矩阵 银河麒麟操作系统 工业4.0 网络结构图 rsyslog OpenManus 博客 服务器数据恢复 数据恢复 存储数据恢复 raid5数据恢复 磁盘阵列数据恢复 visualstudio 远程 命令 执行 sshpass 操作 IDE AI 原生集成开发环境 Trae AI list 模拟实现 驱动开发 硬件工程 嵌入式实习 CLion 远程连接 微信开放平台 微信公众平台 微信公众号配置 图像处理 ArcTS 登录 ArcUI GridItem DigitalOcean GPU服务器购买 GPU服务器哪里有 GPU服务器 vue arkUI 微信小程序域名配置 微信小程序服务器域名 微信小程序合法域名 小程序配置业务域名 微信小程序需要域名吗 微信小程序添加域名 Kylin-Server EasyConnect ollama下载加速 服务器繁忙 Cline Kali Linux 黑客 渗透测试 信息收集 zabbix nextjs reactjs threejs 3D 黑客技术 Open WebUI 流式接口 SenseVoice 本地部署 api openwrt RoboVLM 通用机器人策略 VLA设计哲学 vlm fot robot 视觉语言动作模型 具身智能 ux devops springboot hadoop 大文件分片上传断点续传及进度条 如何批量上传超大文件并显示进度 axios大文件切片上传详细教 node服务器合并切片 vue3大文件上传报错提示错误 大文件秒传跨域报错cors CPU 主板 电源 网卡 压测 ECS ssrf 失效的访问控制 linux 命令 sed 命令 string模拟实现 深拷贝 浅拷贝 经典的string类问题 三个swap 游戏服务器 TrinityCore 魔兽世界 权限 开发环境 SSL证书 npm 半虚拟化 硬件虚拟化 Hypervisor 职场和发展 micropython esp32 mqtt WebUI DeepSeek V3 xrdp minicom 串口调试工具 pygame 小游戏 五子棋 fd 文件描述符 NFS redhat 蓝耘科技 元生代平台工作流 ComfyUI 群晖 文件分享 yum源切换 更换国内yum源 odoo 服务器动作 Server action 雨云服务器 环境迁移 AI写作 prompt 远程控制 远程看看 远程协助 outlook 直播推流 毕设 mq gitea 混合开发 环境安装 JDK ip命令 新增网卡 新增IP 启动网卡 svn 三级等保 服务器审计日志备份 VMware安装Ubuntu Ubuntu安装k8s mysql离线安装 ubuntu22.04 mysql8.0 状态管理的 UDP 服务器 Arduino RTOS yum 5G 3GPP 卫星通信 源码 毕业设计 W5500 OLED u8g2 TCP服务器 matlab jellyfin centos-root /dev/mapper yum clean all df -h / du -sh eureka bootstrap html 京东云 命令行 基础入门 编程 css hive Hive环境搭建 hive3环境 Hive远程模式 Invalid Host allowedHosts openvpn server openvpn配置教程 centos安装openvpn OD机试真题 华为OD机试真题 服务器能耗统计 压力测试 Docker Hub docker pull daemon.json chrome 浏览器下载 chrome 下载安装 谷歌浏览器下载 智能音箱 智能家居 大模型面经 大模型学习 AnythingLLM AnythingLLM安装 transformer 飞牛NAS 飞牛OS MacBook Pro thingsboard dba rime Ubuntu Server Ubuntu 22.04.5 IIS .net core Hosting Bundle .NET Framework vs2022 测试用例 磁盘监控 服务器配置 信号处理 网络药理学 生信 生物信息学 gromacs 分子动力学模拟 MD 动力学模拟 监控k8s集群 集群内prometheus 交叉编译 rdp 实验 王者荣耀 ssh漏洞 ssh9.9p2 CVE-2025-23419 Wi-Fi DNS k8s集群资源管理 云原生开发 RAGFLOW vmware 卡死 iftop 网络流量监控 自动化编程 云电竞 云电脑 todesk 设备 GPU PCI-Express Google pay Apple pay 服务器主板 AI芯片 webstorm Linux无人智慧超市 LInux多线程服务器 QT项目 LInux项目 单片机项目 UOS 统信操作系统 android studio 交互 ros2 moveit 机器人运动 深度优先 图论 并集查找 换根法 树上倍增 telnet 远程登录 代码调试 ipdb 游戏程序 ios dity make 自动驾驶 显示过滤器 ICMP Wireshark安装 ping++ 强制清理 强制删除 mac废纸篓 SWAT 配置文件 服务管理 网络共享 Ark-TS语言 gaussdb 实战案例 DeepSeek行业应用 Heroku 网站部署 k8s资源监控 annotations自动化 自动化监控 监控service 监控jvm 视觉检测 游戏机 xml VMware创建虚拟机 思科模拟器 Cisco sqlite3 Docker Compose docker-compose saltstack 虚拟显示器 AI作画 聊天室 openstack Xen 信创 信创终端 中科方德 RAG 检索增强生成 文档解析 大模型垂直应用 AI agent Playwright 自动化测试 nac 802.1 portal QT 5.12.12 QT开发环境 Ubuntu18.04 Radius wordpress 无法访问wordpess后台 打开网站页面错乱 linux宝塔面板 wordpress更换服务器 bcompare Beyond Compare 健康医疗 互联网医院 模拟器 教程 qt项目 qt项目实战 qt教程 muduo ros 国标28181 视频监控 监控接入 语音广播 流程 SIP SDP Ubuntu共享文件夹 共享目录 Linux共享文件夹 edge浏览器 物联网开发 3d 社交电子 技能大赛 RAGFlow ROS EMQX 通信协议 聚类 gcc cursor 计算虚拟化 弹性裸金属 程序 性能分析 yaml Ultralytics 可视化 SEO chfs ubuntu 16.04 小番茄C盘清理 便捷易用C盘清理工具 小番茄C盘清理的优势尽显何处? 教你深度体验小番茄C盘清理 C盘变红?!不知所措? C盘瘦身后电脑会发生什么变化? neo4j 知识图谱 虚拟局域网 显示管理器 lightdm gdm ceph 英语 vscode 1.86 强化学习 xcode EMUI 回退 降级 升级 gnu langchain 直流充电桩 充电桩 GCC aarch64 编译安装 HPC 数据库架构 数据管理 数据治理 数据编织 数据虚拟化 idm 深度求索 私域 知识库 本地部署AI大模型 LInux 显卡驱动 火绒安全 Nuxt.js apt 大模型应用 HTTP 服务器控制 ESP32 DeepSeek 域名服务 DHCP 符号链接 配置 备选 网站 调用 示例 AD域 SSH 密钥生成 SSH 公钥 私钥 生成 tcp jina 匿名管道 frp seatunnel jvm seleium chromedriver Linux的基础指令 vasp安装 边缘计算 图形化界面 composer AutoDL iperf3 带宽测试 ruby 游戏引擎 crosstool-ng 计算机 鲲鹏 昇腾 npu 双系统 GRUB引导 Linux技巧 业界资讯 模拟退火算法 tensorflow 高级IO epoll 银河麒麟高级服务器 外接硬盘 Kylin HarmonyOS 技术共享 glibc 华为机试 数据库系统 自定义客户端 SAS ubuntu24.04.1 软负载 AI-native Docker Desktop HiCar CarLife+ CarPlay QT RK3588 yolov8 SysBench 基准测试 回显服务器 UDP的API使用 web milvus 做raid 装系统 armbian u-boot 搜索引擎 elk VM搭建win2012 win2012应急响应靶机搭建 攻击者获取服务器权限 上传wakaung病毒 应急响应并溯源 挖矿病毒处置 应急响应综合性靶场 fast ip 信号 minio 链表 rnn ftp CVE-2024-7347 内网服务器 内网代理 内网通信 uniapp web3 Dell HPE 联想 浪潮 vscode1.86 1.86版本 ssh远程连接 sublime text big data MacMini 迷你主机 mini Apple opensearch helm 宠物 免费学习 宠物领养 宠物平台 历史版本 下载 etl 单一职责原则 运维监控 open webui IMM 代理 spark HistoryServer Spark YARN jobhistory VR手套 数据手套 动捕手套 动捕数据手套 asp.net大文件上传下载 VSCode 移动云 程序员创富 数学建模 zookeeper nfs 服务器部署ai模型 SSL 域名 cd 目录切换 Anolis nginx安装 linux插件下载 输入法 av1 电视盒子 机顶盒ROM 魔百盒刷机 c/c++ 串口 宝塔 指令 sonoma 自动更新 文件系统 用户缓冲区 小智AI服务端 xiaozhi ASR TTS x64 SIGSEGV SSE xmm0 xshell termius iterm2 GIS 遥感 WebGIS 策略模式 单例模式 miniapp 真机调试 调试 debug 断点 网络API请求调试方法 数据仓库 数据库开发 database 支付 微信支付 开放平台 AD 域管理 网站搭建 serv00 大大通 第三代半导体 碳化硅 ArkTs ArkUI Windows ai工具 v10 软件 ldap keepalived xpath定位元素 多个客户端访问 IO多路复用 TCP相关API curl wget cocoapods h.264 chrome devtools URL MI300x 集成学习 毕昇JDK 上传视频至服务器代码 vue3批量上传多个视频并预览 如何实现将本地视频上传到网页 element plu视频上传 ant design vue vue3本地上传视频及预览移除 Kali 渗透 open Euler dde LLM Web APP Streamlit 设置代理 实用教程 网卡的名称修改 eth0 ens33 Cursor 网工 版本 Ubuntu DeepSeek DeepSeek Ubuntu DeepSeek 本地部署 DeepSeek 知识库 DeepSeek 私有化知识库 本地部署 DeepSeek DeepSeek 私有化部署 SRS 流媒体 直播 相机 SSH 服务 SSH Server OpenSSH Server Deepseek-R1 私有化部署 推理模型 自动化任务管理 maxkb ARG easyui vue-i18n 国际化多语言 vue2中英文切换详细教程 如何动态加载i18n语言包 把语言json放到服务器调用 前端调用api获取语言配置文件 nlp OpenHarmony python3.11 黑苹果 视频编解码 飞牛nas fnos 推荐算法 sdkman Ubuntu 24.04.1 轻量级服务器 sequoiaDB dash 正则表达式 pgpool 端口测试 性能测试 田俊楠 图形渲染 Ubuntu22.04 开发人员主页 trea idea 常用命令 文本命令 目录命令 崖山数据库 YashanDB 远程过程调用 Windows环境 Xinference prometheus数据采集 prometheus数据模型 prometheus特点 高效日志打印 串口通信日志 服务器日志 系统状态监控日志 异常记录日志 UOS1070e alias unalias 别名 相差8小时 UTC 时间 netty DocFlow 无桌面 risc-v ubuntu24 vivado24 键盘 Jellyfin 免费域名 域名解析 swoole docker命令大全 FTP服务器 怎么卸载MySQL MySQL怎么卸载干净 MySQL卸载重新安装教程 MySQL5.7卸载 Linux卸载MySQL8.0 如何卸载MySQL教程 MySQL卸载与安装 rustdesk 本地化部署 embedding 考研 软考 嵌入式系统开发 代理服务器 Linux的权限 联想开天P90Z装win10 私有化 deep learning 宕机切换 服务器宕机 监控k8s 监控kubernetes 执法记录仪 智能安全帽 smarteye flash-attention 互信 ecm bpm Linux awk awk函数 awk结构 awk内置变量 awk参数 awk脚本 awk详解 Ubuntu 24 常用命令 Ubuntu 24 Ubuntu vi 异常处理 Minecraft DOIT 四博智联 Claude 北亚数据恢复 oracle数据恢复 实习 c 远程服务 MCP server C/S windows日志 MySql conda配置 conda镜像源 XCC Lenovo 反向代理 cnn DenseNet 稳定性 看门狗 繁忙 解决办法 替代网站 汇总推荐 AI推理 CrewAI 邮件APP 免费软件 H3C iDRAC R720xd qemu libvirt 大模型部署 freebsd can 线程池 金仓数据库 2025 征文 数据库平替用金仓 skynet 增强现实 沉浸式体验 应用场景 技术实现 案例分析 AR 服务器无法访问 ip地址无法访问 无法访问宝塔面板 宝塔面板打不开 IM即时通讯 剪切板对通 HTML FORMAT XFS xfs文件系统损坏 I_O error es GoogLeNet 前后端分离 沙盒 uni-file-picker 拍摄从相册选择 uni.uploadFile H5上传图片 微信小程序上传图片 FunASR file server http server web server 阿里云ECS USB网络共享 云桌面 微软 AD域控 证书服务器 LORA NLP 个人博客 deepseek r1 X11 Xming ssh远程登录 虚幻引擎 make命令 makefile文件 我的世界 我的世界联机 数码 iphone centos 7 docker run 数据卷挂载 交互模式 ISO镜像作为本地源 镜像 selete 阻塞队列 生产者消费者模型 服务器崩坏原因 jetty undertow grub 版本升级 扩容 less linux环境变量 磁盘镜像 服务器镜像 服务器实时复制 实时文件备份 Erlang OTP gen_server 热代码交换 事务语义 powerpoint 路径解析 MNN Qwen 环境配置 备份SQL Server数据库 数据库备份 傲梅企业备份网络版 音乐服务器 Navidrome 音流 eNSP 网络规划 VLAN 企业网络 tidb GLIBC dns是什么 如何设置电脑dns dns应该如何设置 银河麒麟桌面操作系统 Kylin OS MQTT协议 消息服务器 代码 Attention searxng 在线预览 xlsx xls文件 在浏览器直接打开解析xls表格 前端实现vue3打开excel 文件地址url或接口文档流二进 PPI String Cytoscape CytoHubba Docker引擎已经停止 Docker无法使用 WSL进度一直是0 镜像加速地址 perf 人工智能生成内容 pppoe radius DBeaver kerberos hugo Netty 即时通信 NIO TCP协议 搭建个人相关服务器 抗锯齿 nftables 防火墙 word 分布式训练 多路转接 软件构建 firewall deekseek cfssl Typore 大模型技术 本地部署大模型 Logstash 日志采集 计算生物学 生物信息 基因组 HarmonyOS NEXT 原生鸿蒙 MVS 海康威视相机 根服务器 clickhouse 项目部署 HAProxy junit IDEA 上传视频文件到服务器 uniApp本地上传视频并预览 uniapp移动端h5网页 uniapp微信小程序上传视频 uniapp app端视频上传 uniapp uview组件库 lb 协议 嵌入式Linux IPC Linux环境 compose laravel dock 加速 中兴光猫 换光猫 网络桥接 自己换光猫 OpenSSH 需求分析 规格说明书 欧标 OCPP 多端开发 智慧分发 应用生态 鸿蒙OS lua 容器技术 IPv4 子网掩码 公网IP 私有IP 音乐库 飞牛 主从复制 react native 元服务 应用上架 查询数据库服务IP地址 SQL Server 语音识别 K8S k8s管理系统 DIFY 虚拟现实 SVN Server tortoise svn wpf 智能电视 EtherNet/IP串口网关 EIP转RS485 EIP转Modbus EtherNet/IP网关协议 EIP转RS485网关 EIP串口服务器 flink yolov5 eclipse 自学笔记 小米 澎湃OS Android IO模型 端口 查看 ss deployment daemonset statefulset cronjob 读写锁 AI Agent 字节智能运维 项目部署到linux服务器 项目部署过程 autodl win服务器架设 windows server llama.cpp g++ g++13 sysctl.conf vm.nr_hugepages IMX317 MIPI H265 VCU k8s二次开发 集群管理 Headless Linux onlyoffice 状态模式 vpn DeepSeek r1 热榜 AzureDataStudio 语法 7z Qwen2.5-VL vllm VS Code 僵尸世界大战 游戏服务器搭建 WSL2 上安装 Ubuntu NLP模型 PX4 java-rocketmq 网络建设与运维 hexo 服务网格 istio bat 内网环境 Mac内存不够用怎么办 架构与原理 banner 进程优先级 调度队列 进程切换 Doris搭建 docker搭建Doris Doris搭建过程 linux搭建Doris Doris搭建详细步骤 Doris部署 离线部署dify fonts-noto-cjk 我的世界服务器搭建 minecraft docker部署翻译组件 docker部署deepl docker搭建deepl java对接deepl 翻译组件使用 cpp-httplib GameFramework HybridCLR Unity编辑器扩展 自动化工具 Unity Dedicated Server Host Client 无头主机 鸿蒙开发 移动开发 企业网络规划 华为eNSP windows 服务器安装 kernel 捆绑 链接 谷歌浏览器 youtube google gmail su sudo sentinel 区块链 midjourney opengl 代码托管服务 css3 regedit 开机启动 佛山戴尔服务器维修 佛山三水服务器维修 存储维护 NetApp存储 EMC存储 TrueLicense webgl 加解密 Yakit yaklang perl 干货分享 黑客工具 密码爆破 在线office 李心怡 超融合 影刀 #影刀RPA# docker部署Python 像素流送api 像素流送UE4 像素流送卡顿 像素流送并发支持 玩机技巧 软件分享 软件图标 Python基础 Python教程 Python技巧 axure 富文本编辑器 tailscale derp derper 中转 triton 模型分析 WLAN 线性代数 电商平台 服务器时间 C++软件实战问题排查经验分享 0xfeeefeee 0xcdcdcdcd 动态库加载失败 程序启动失败 程序运行权限 标准用户权限与管理员权限 linux内核 NAT转发 NAT Server Mermaid 可视化图表 软链接 硬链接 移动魔百盒 ubuntu20.04 开机黑屏 USB转串口 harmonyOS面试题 安防软件 CDN 安卓模拟器 figma WebVM 基础环境 流水线 脚本式流水线 ArtTS Zoertier 内网组网 本地环回 bind iventoy VmWare OpenEuler virtualbox 浏览器开发 AI浏览器 Spring Security rtsp服务器 rtsp server android rtsp服务 安卓rtsp服务器 移动端rtsp服务 大牛直播SDK proxy模式 游戏开发 烟花代码 烟花 元旦 本地知识库部署 DeepSeek R1 模型 带外管理 lsb_release /etc/issue /proc/version uname -r 查看ubuntu版本 rancher 磁盘清理 navicat 大模型推理 网络搭建 神州数码 神州数码云平台 云平台 xss hosts db 机柜 1U 2U top Linux top top命令详解 top命令重点 top常用参数 输入系统 stable diffusion React Next.js 开源框架 云耀服务器 ubuntu 18.04 fstab Apache Beam 批流统一 案例展示 数据分区 容错机制 ros1 Noetic 20.04 apt 安装 联机 僵尸毁灭工程 游戏联机 开服 powerbi 网络爬虫 netlink libnl3 docker搭建pg docker搭建pgsql pg授权 postgresql使用 postgresql搭建 开源软件 话题通信 服务通信 对比 工具 meld DiffMerge 服务器安全 网络安全策略 防御服务器攻击 安全威胁和解决方案 程序员博客保护 数据保护 安全最佳实践 达梦 DM8 镜像下载 notepad oracle fusion oracle中间件 硅基流动 ChatBox IO 隐藏文件 隐藏目录 管理器 通配符 Alexnet tar WebServer 生活 风扇控制软件 ROS2 图片增强 增强数据 mysql安装报错 windows拒绝安装 软件卸载 系统清理 智慧农业 开源鸿蒙 团队开发 跨平台 AI员工 国产数据库 瀚高数据库 数据迁移 下载安装 cron crontab日志 Linux权限 权限命令 特殊权限 服务器扩容没有扩容成功 rpa CPU 使用率 系统监控工具 linux 命令 端口聚合 windows11 视频平台 录像 视频转发 视频流 MAC SecureCRT Linux find grep 钉钉 授时服务 北斗授时 UEFI Legacy MBR GPT U盘安装操作系统 抓包工具 System V共享内存 进程通信 MacOS 服务器部署 本地拉取打包 nosql docker desktop image qt5 客户端开发 visual studio ELF加载 Web服务器 多线程下载工具 PYTHON 合成模型 扩散模型 图像生成 cmake 浪潮信息 AI服务器 MDK 嵌入式开发工具 论文笔记 Web应用服务器 CosyVoice 华为证书 HarmonyOS认证 华为证书考试 MAVROS 四旋翼无人机 fork wait waitpid exit ABAP python2 ubuntu24.04 kotlin Carla 智能驾驶 显示器 csrutil mac恢复模式进入方法 恢复模式 内核 ranger MySQL8.0 Helm k8s集群 极限编程 VMware Tools vmware tools安装 vmwaretools安装步骤 vmwaretools安装失败 vmware tool安装步骤 vm tools安装步骤 vm tools安装后不能拖 vmware tools安装步骤 安装MySQL vnc samba 流量运营 通信工程 毕业 iNode Macos yum换源 多产物 zip unzip Reactor反应堆 ebpf uprobe Linux 维护模式 网络文件系统 mcp服务器 client close 代码规范 scapy lvm 磁盘挂载 磁盘分区 电视剧收视率分析与可视化平台 Unity插件 wsgiref Web 服务器网关接口 查看显卡进程 fuser xfce 浏览器自动化 nvm whistle 问题解决 java-rabbitmq ardunio BLE 粘包问题 Sealos UDP 论文阅读 性能调优 安全代理 scikit-learn burp suite 抓包 vu大文件秒传跨域报错cors 网页服务器 web服务器 Nginx mm-wiki搭建 linux搭建mm-wiki mm-wiki搭建与使用 mm-wiki使用 mm-wiki详解 copilot Mac软件 ip协议 NFC 近场通讯 智能门锁 shell脚本免交互 expect linux免交互 export import save load 迁移镜像 宝塔面板无法访问 大屏端 ubuntu安装 linux入门小白 录音麦克风权限判断检测 录音功能 录音文件mp3播放 小程序实现录音及播放功能 RecorderManager 解决录音报错播放没声音问题 搜狗输入法 中文输入法 笔灵AI AI工具 数字证书 签署证书 工具分享 服务器正确解析请求体 解决方案 接口优化 js lighttpd安装 Ubuntu配置 Windows安装 服务器优化 MobaXterm 弹性服务器 Docker快速入门 DevOps 软件交付 数据驱动 联网 easyconnect WINCC 服务器ssl异常解决 文件传输 Chatbox 卷积神经网络 ShapeFile GeoJSON 零售 西门子PLC 通讯 蓝牙 macOS zerotier