最新资讯

  • Linux-基础IO

Linux-基础IO

2025-05-05 22:01:05 6 阅读

🌎Linux基础IO


文章目录:

Linux基础IO

    C语言中IO交互
      常用C接口
        fopen
        fputs
        fwrite
        fgets

      当前路径
      三个文件流

    系统文件IO
      open函数
      参数含义
      close函数

      write函数
        参数含义

    文件描述符fd
      认识文件描述符

      重定向
        输出重定向
        输入重定向
        追加重定向
        重定向接口

    缓冲区
      简单认识缓冲区
      技术角度认识缓冲区
      FILE结构体
      编码模拟

    总结


前言:

  在刚开始学习Linux的时候,我们记住了Linux下一切皆文件,我们通过这篇文章来深入了解Linux下文件的构成及应用。


🚀C语言中IO交互

✈️ 常用C接口
🚩 fopen

fopen:打开一个文件。

代码示例:

#include

int main()
{
    FILE* fp = fopen("./log.txt", "w");//打开一个文件,如果没有则创建一个文件
    if(fp == NULL)
    {
        perror("fopen");
        return 1;
    }

	//文件操作介于打开和关闭之间

    fclose(fp);//关闭文件
    return 0;
}

注意

  当以 ‘w’ 方式打开文件时:该文件会被清空。

  当以 ‘a’ 方式打开文件时:正常打开该文件,如果有写入操作则是追加写入。

  当以 ‘r’ 方式打开文件时:仅读取文件。


🚩 fputs

fputs:向文件流中写入一个字符串

代码示例:

#include

int main()
{
    FILE* fp = fopen("./log.txt", "w");
    if(fp == NULL)
    {
        perror("fopen");
        return 1;
    }

    const char* str = "this is file operate
";
    fputs(str, fp);

    fclose(fp);
    return 0;
}


🚩 fwrite

fwrite:向二进制文件写入数据。

代码示例:

#include
#include

#define FILENAME "log.txt"

int main()
{
    FILE* fp = fopen("./log.txt", "w");
    if(fp == NULL)
    {
        perror("fopen");
        return 1;
    }

    const char* msg = "this is file operate
";
    int cnt = 5;
    while(cnt)
    {
        fwrite(msg, strlen(msg), 1, fp);
        printf("write %d block
", n);
        cnt--;

    }

    fclose(fp);
    return 0;
}

第一个参数:

写入数据的对象。

第二个参数:

基本单位的大小。

第三个参数:

表示写入多少个基本单位。

第四个参数:

表示文件流

返回值:

表示写入的基本单位的个数,也就是第三个参数


🚩 fgets

fgets:读取一个字符串。

代码示例:

#include
#include
#include
#include

int main()
{
    FILE* fp = fopen("./log.txt", "r");//r方式打开
    if(fp == NULL)
    {
        perror("fopen");
        return 1;
    }

    char buffer[64];
    while(1)
    {
        char* r = fgets(buffer, sizeof(buffer), fp);
        if(!r) break;

        printf("%s
", buffer);
    }

    fclose(fp);
    return 0;
}

  这里我只列举了部分常用C语言IO接口,如果有遗忘,请自行复习。


✈️当前路径

  当我们在程序中创建一个文件时,例如使用 fopen函数以 ‘w’ 方式打开文件,文件不存在时则创建文件,但是为什么文件创建位置是在当前路径下呢?


  其实是通过该进程的一项属性数据来判断所处路径的,我们可以查询该进程pid,在proc目录下进行查看该进程:

  cwd表示该进程当前所处工作目录,exe表示可执行程序所处路径。

注意: 当前路径不是指可执行程序所处路径,而是指该程序运行为进程时所处路径。


✈️三个文件流

  刚开始接触Linux的时候,我们都知道有句话叫做:Linux下一切皆文件,那么键盘、显示器、网卡、声卡等等这些对于Linux来说都是文件!

  我们使用Linux都知道,想要对一个文件进行操作,我们必须要打开一个文件,这是必须的。但是为什么 显示器文件键盘文件 这些文件我们并不需要直接打开就可以直接使用呢?

文件在打开的前提一定是基于进程的,而进程在运行的过程中会打开默认的三个流,即标准输入流,标准输出流、标准错误流。而对应C语言中就是 stdin、stdout、stderr

  标准输入流对应的设备是键盘、标准输出与标准错误流对应的设备是显示器。


  当我们使用C语言运行一个程序的时候,操作系统会默认将这三个流给打开,于是,我们使用printf、scanf、gets、puts等接口时可以直接使用。

  也就是说我们的输入输出是因为stdin和stdout流是默认打开的状态,我们可以根据stdin、stdout来直接对屏幕进行输出:

#include

int main()
{
    fprintf(stdout, "you can see me
");//对标准输出流进行写入
    fprintf(stdout, "yes I'can
");//对标准输出流进行写入
    return 0;
}


  对标准输出流进行写入,其实就是将数据打印到显示器上!

注意:并不是只有C语言有此特性,其他语言例如C++的cout、cin也具有标准流。这种特性并不是有语言层面提供的,而是由操作系统提供的。


🚀系统文件IO

  除了使用C语言或者其他语言的IO交互,我们也可以采用调用系统接口来进行文件访问,而系统调用时更接近于底层的,其他语言都是对系统的系统调用进行封装的。

✈️open函数

open函数是fopen函数的底层,其为Linux的系统调用,函数原型为:

int open(const char *pathname, int flags, mode_t mode);
参数含义
  • pathname:表示 需要传入的文件路径,当只有文件名的时候,表示子在当前目录打开或创建该文件。

  • flags:表示打开文件的方式。通常打开文件的常用方式分为以下几种:

flags选项含义
O_RDONLY以只读的方式打开文件
O_WRONLY以只写的方式打开文件
O_APPEND以追加的方式打开文件
O_CREAT文件不存在时,则创建文件
O_RDWR以读写的方式打开文件
O_TRUNC清空文件
  • mode:表示创建文件的默认方式。不需要创建文件时,这个参数不必传参。

  为了能理解第二个参数flags ,我们通过以下代码来观察:

#include

#define O_LISTEN 1// 0001
#define O_TALK 2 // 0010
#define O_READ 4  // 0100
#define O_WRITE 8 // 1000

void Listen()
{
    printf("linten English dialog
");
}

void Talk()
{
    printf("talk about English
");
}

void Read()
{
    printf("read English newspaper
");
}

void Write()
{
    printf("write English article
");
}

void operate(int flags)
{
	//根据二进制位来判断调用函数接口类型
    if(flags & O_LISTEN)
        Listen();
    if(flags & O_TALK)
        Talk();
    if(flags & O_READ)
        Read();
    if(flags & O_WRITE)
        Write();
}

int main()
{
    operate(O_LISTEN);
    printf("
");
    operate(O_TALK | O_READ);//按位或运算调用
    printf("
");
    operate(O_LISTEN | O_TALK | O_READ | O_WRITE);
    return 0;
}

  我们可以 使用或运算 来做出 不同的行为,同样,open接口的flags参数也是如此使用方式,例如,我们以 使用open模拟fopen函数的 ‘w’ 行为

#include
#include
#include
#include

int main()
{
    int fd = open("log.txt", O_WRONLY|O_CREAT, 0666);//文件默认权限设置为666
    if(fd == -1)
    {
        perror("open");
    }
    return 0;
}

  我们确实模仿出了fopen函数的功能,仔细看文件权限,与我们想要的并不同,最后三项应该是 rw- 才对,这是因为存在叫做 权限掩码(umask) 的东西,其通常默认为0002,与mode的关系是 umask & mode,所以我们在设置权限之前,需要把umask设置为0:

#include
#include
#include
#include

int main()
{
	umask(0);
    int fd = open("log.txt", O_WRONLY|O_CREAT, 0666);//文件默认权限设置为666
    if(fd == -1)
    {
        perror("open");
    }
    return 0;
}

✈️close函数
int close(int fd);

  close函数属于Linux下的系统调用,其功能是 关闭一个文件描述符,参数是 有待关闭的文件描述符。

✈️write函数

函数定义

ssize_t write(int fd, const void* buf, size_t count);
🚩 参数含义
  • fd:需要传入的文件描述符。
  • buf:需要写入的字符串的起始位置。
  • count:需要写入字符串的长度。

  其中第三个参数需要注意,传入的字符串长度是不算 的,因为这是系统调用接口,并非C语言。

#include
#include
#include
#include
#include
#include

int main()
{
    umask(0);
    int fd = open("log.txt", O_WRONLY|O_CREAT, 0666);
    if(fd == -1)
    {
        perror("open");
        return 1;
    }

    const char* str = "hello sys call
";
    write(fd, str, strlen(str));//长度不算 

    close(fd);
    return 0;
}

  但是如果我们写入的字符串改变了并且没有 :

const char* str = "aaaa";
write(fd, str, strlen(str));


  如果这样,那么下次进行写入就是以 覆盖的方式进行写入。所以我们在打开文件的时候需要将open函数的选项增加一个 O_TRUNC 选项:

int fd = open("log.txt", O_WRONLY|O_CREAT|O_TRUNC, 0666);

  如果需要实现什么功能,就需要提供对应的选项。


🚀文件描述符fd

  文件描述符在上文中不止出现了一次,包括 open 函数的返回值,close 函数的参数等等,从其出现的频率来看,似乎是很重要的一个东西。

✈️认识文件描述符

  既然open 函数返回值是文件描述符,那么我们可以创建多个open函数,使用多个返回值接收并且打印来观察现象:

#include
#include
#include
#include
#include
#include

int main()
{
    int fd1 = open("log1.txt", O_WRONLY|O_CREAT|O_TRUNC, 0666);
    int fd2 = open("log2.txt", O_WRONLY|O_CREAT|O_TRUNC, 0666);
    int fd3 = open("log3.txt", O_WRONLY|O_CREAT|O_TRUNC, 0666);
    int fd4 = open("log4.txt", O_WRONLY|O_CREAT|O_TRUNC, 0666);
    int fd5 = open("log5.txt", O_WRONLY|O_CREAT|O_TRUNC, 0666);

    printf("fd1:%d
", fd1);
    printf("fd2:%d
", fd2);
    printf("fd3:%d
", fd3);
    printf("fd4:%d
", fd4);
    printf("fd5:%d
", fd5);
    return 0;
}

  我们观察到的现象是,文件描述符是从3开始的,那么012去哪里了?并且为什么它们是连续的??

  其实0、1、2文件描述符已经被使用了!其分别是:标准输入、标准输出、标准错误!而它们是连续的,其实也就是 数组下标

  而我们在上文中也提到过三个标准流,即:

  他们的类型都是 FILE* 类型,其实 FILE 是C标准库自己封装的一个结构体。而这三个流分别是文件描述符的前三个,那么 FILE 结构体内必定 封装特定的fd!

  我们经常说,Linux下一切皆文件,那么一个空文件,它的大小真的是0吗,其实在很久以前我们也探讨过,只要文件被创建,那么就不可能为0。

  文件 = 内容 + 属性

  那么每个文件必然具有一些相同的初始属性,比如文件标志位,文件权限位,文件对下一个文件的指针,缓冲区等等。这些属性很杂乱,所以操作系统需要对其进行管理,那么还是那六个字:先描述,再组织

  将这些属性组织到结构体当中,便更有利于操作系统的管理:

  在task_struct 中存在一个 files 指针,该指针指向一个 files_struct 的结构体,在该结构体当中存在一个 fd_array 的指针数组,而 数组的下标就对应我们所谓的文件描述符

  因为0、1、2这三个文件描述符时默认打开的,但是这里我把它关闭(仅关闭0位置),再使用 open 创建一个文件,会发生什么?

#include
#include
#include
#include
#include
#include

int main()
{
    close(0);
    int fd1 = open("log1.txt", O_WRONLY|O_CREAT|O_TRUNC, 0666);
    int fd2 = open("log2.txt", O_WRONLY|O_CREAT|O_TRUNC, 0666);
    int fd3 = open("log3.txt", O_WRONLY|O_CREAT|O_TRUNC, 0666);
    int fd4 = open("log4.txt", O_WRONLY|O_CREAT|O_TRUNC, 0666);
    int fd5 = open("log5.txt", O_WRONLY|O_CREAT|O_TRUNC, 0666);

    printf("fd1:%d
", fd1);
    printf("fd2:%d
", fd2);
    printf("fd3:%d
", fd3);
    printf("fd4:%d
", fd4);
    printf("fd5:%d
", fd5);
    return 0;
}


  0位置的fd被我们关闭了,但当我们在创建文件的时候,0号位置被新创建的文件占用了。如果我再关闭2号文件描述符呢?

close(0);
close(2);

  看来我们 关闭 一个默认打开的文件描述符,那么新建文件就会:按照顺序占用 被关闭(未被使用) 的文件描述符。


✈️重定向

  了解了什么是文件描述符之后,我们就可以根据文件描述符的规则来实现不同的重定向功能。

  我们在最开始学习Linux指令的时候使用过重定向功能,而重定向无外乎 输入重定向输出重定向

  重定向的原理是,将原本需要输入或者输出的对象文件变为指定的对象文件

  比如,我们知道Linux下一些皆文件,那么键盘、显示器都是文件,而我们平常的打印,其实就是对 “显示器文件” 上进行写入,而重定向就是将原本向 “显示器文件” 写入更改为向其他文件写入。

🚩 输出重定向

  而更改重定向文件其实是就是更改文件描述符指向的文件:

我们使用C语言来模拟一下情况:

#include
#include
#include
#include
#include
#include
#include

int main()
{
    close(1);
    umask(0);
    int fd = open("log1.txt", O_WRONLY|O_CREAT, 0666);
    if(fd < 0)
    {
        perror("open");
        return 1;
    }

    printf("Hello Linux xxxxxxxxxxxx
");
    printf("Hello Linux xxxxxxxxxxxx
");
    printf("Hello Linux xxxxxxxxxxxx
");
    printf("Hello Linux xxxxxxxxxxxx
");

    fflush(stdout);

    close(fd);
    return 0;
}


🚩 输入重定向

  同样,输入重定向也是先关闭默认打开的0号文件描述符,使得新创建的分配到0号文件描述符,这样进行输入的时候就重定向到该文件内:

C语言模拟:

#include
#include
#include
#include
#include
#include
#include

int main()
{
    close(0);
    umask(0);
    int fd = open("log1.txt", O_RDONLY|O_CREAT, 0666);//这里为只读
    if(fd < 0)
    {
        perror("open");
        return 1;
    }

    char buff[64];
    while(scanf("%s", buff) == EOF)
    {
		printf("%s
", buff);
	}

    close(fd);
    return 0;
}


🚩 追加重定向

  追加重定向,与输出重定向不同的是,输出重定向每次向文件内输入时都会清空文件内容再做输入,而追加重定向是追加写入文件内,不修改原来文件的文本。

  其实实现起来也很简单,将open 函数的flags参数添加上 O_APPEND 即可:

#include
#include
#include
#include
#include
#include
#include

int main()
{
    close(1);
    umask(0);
    int fd = open("log1.txt", O_WRONLY|O_CREAT|O_APPEND, 0666);
    if(fd < 0)
    {
        perror("open");
        return 1;
    }

    printf("Hello Linux xxxxxxxxxxxx
");
    printf("Hello Linux xxxxxxxxxxxx
");
    printf("Hello Linux xxxxxxxxxxxx
");
    printf("Hello Linux xxxxxxxxxxxx
");

    fflush(stdout);

    close(fd);
    return 0;
}


🚩 重定向接口

  我们整个重定向需要搞那么麻烦吗?万一在代码段当中添加了其他需求到最后自己是否还能认得这段代码?为了方便,Linux给我们提供了一个接口,dup2

直接一段代码来看用法:

#include
#include
#include
#include
#include
#include
#include

int main()
{
    umask(0);
    int fd = open("log.txt", O_WRONLY|O_CREAT|O_TRUNC, 0666);
    dup2(fd, 1);//将fd重定向到1

    printf("Can you see me??
");
    printf("Can you see me??
");
    printf("Can you see me??
");
    printf("Can you see me??
");

    close(fd);
    return 0;
}

  追加与输入重定向皆可使用dup接口进行重定向,这样简化了代码量,使代码更具可读性。


🚀缓冲区

✈️简单认识缓冲区

  我们可能经常听到 “缓冲区” 这个词,它到底是什么或许你还没有深究过,缓冲区本质上就是一块内存区域,那么为什么要有缓冲区呢?我们来看下面的例子:

  他穿越了,穿越到了千禧年的大学生身上,有一个高中同学叫做阿飞,阿熊在安徽上大学,阿飞在广东上大学。过两天就是阿飞的生日,阿熊买了一个最新显卡准备坐火车送给阿飞,于是阿熊买了火车票,一路颠簸的去掉了广东,然后把礼物送给阿飞,吃顿饭就走了。

  阿熊回到了2024年,正巧阿熊现在的高中同学阿乐也在广东准备过生日,阿熊在安徽,于是阿熊精心挑选了一个键盘,准备送给阿乐。现在是2024年,阿熊拿上键盘直接下楼到邮政快递公司把快递寄了过去。
  但是快递公司并不是拿到你的快递就开始出发配送,而是要等到一定的量到了再配送。等到一车货够了,那么就会出发送快递。过了两天阿乐收到了你的消息,于是也下楼到邮政取了快递。

  上述情况,我们仅仅是为了送一个生日礼物,但是这样的开销是不是就太大了,不仅要买来回车票,到地方可能还要住旅馆,而且花费的时间也很多。这就好比操作系统把每一次的输入都立即送到显示器上一样,电信号看似很快,但是千千万万个信息呢?

  而有了快递公司就方便了许多,只需要下楼寄个快递,等到一定数目的快递集齐了快递就可以发过去了,而对方收到快递也仅仅只需要下楼到快递公司取个快递。

  不论是C语言,还是操作系统,它们同样如此,既然一次一次来回写入开销很大,倒不如开辟一块内存区域,当内容空间的内容满了,再做刷新。

  所以,总的来说,缓冲区其实就是 以空间换时间的一种方式


✈️技术角度认识缓冲区

  我们以前所接触的缓冲区几乎都是语言层面的缓冲区,而缓冲区也分为系统层和语言层缓冲区。

  C语言中的printf/fgets等函数底层其实就是调用系统调用来实现输出的。但是系统调用本身就是需要成本的,所以我们用户层面就要尽量较少的访问系统调用。

  这就好比,阿熊月末没钱了,通常一顿饭要10块钱,撑到下个月大概还有不到10顿,那么阿熊是向朋友一次借10块分10次借还是一次借100就借一次呢?显然阿熊会选择后者。

  C语言也是这么想的,所以C原也提供了缓冲区,我们通常写入数据其实 写入的是C语言的缓冲区,再由C语言调用系统调用把数据刷新到内核当中。从而间接减少系统调用的次数。

缓冲类型分为:

  • 全缓冲:全部刷新,普通文件缓冲区写满才刷新。
  • 行刷新: 之前的内容进行刷新。
  • 无刷新:无刷新。

✈️FILE结构体

  既然存在缓冲区这个东西,那么它存储在哪呢?实际上 缓冲区是由FILE结构体来维护的

  在上文我们说stdin、stdout、stderr这三个流的类型皆是 FILE* 类型,而每个文件都有自己的FILE结构体,所以 每个文件都有自己的缓冲区

  不仅如此,C语言的很多接口的参数也都是FILE* 类型:

   拿fwrite来举例,仅仅是把 *ptr 的 (size * nmemb) 字节大小的内容拷贝到 FILE 缓冲区内,需要的时候内部再决定如何刷新。

  所以这些接口大部分时间都是向FILE内的缓冲区进行拷贝,所以在 用户层面上这些接口的效率也比较高

我们来看看C语言库是如何定义的:

/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
};

  由此可以清晰的观察到C语言级别的缓冲区是如何定义的,这里需要注意的另一个点是 文件描述符被 _fileno 封装

  下面我写一段代码来证明缓冲区的存在:

#include
#include
#include

int main()
{
    //Use system call
    const char* s1 = "hello write
";
    write(1, s1, strlen(s1));

    //Use C inteface
    const char* s2 = "hello fprintf
";
    fprintf(stdout, "%s", s2);

    const char* s3 = "hello  fwrite
";
    fwrite(s3, strlen(s3), 1, stdout);

    fork();//注意这里进行fork
    return 0;
}


  这个现象就很有趣了,第一次运行没什么问题,三个数据全部打印出来,但是当我们第二次运行并且重定向到空文件当中时却出了问题,你可以先思考为什么。

  其实这是因为,第一次运行程序其实是向显示器打印,这个行为默认的刷新行为是 行刷新。而第二次重定向到了文件中,这个时候刷新方式就变为了 全缓冲
  而全缓冲正常情况下是进程退出时才进行刷新策略的。而在程序的最后我们进行了fork创建了子进程。
  而这个时候,缓冲区接收的数据没有满,所以这个时候不论哪个进程先退出,都会将数据写入到C语言中的缓冲区当中,最终造成了打印出来的数据有两项是重复的。
  而write为什么只打印一次?这是因为write函数是系统调用,并 不参与 语言层的缓冲区,所以只打印一次。

  当某一个进程退出时,那么一定要将自己缓冲区中的数据刷新到内核当中,而 刷新的本质就是写入!而一旦写入就会 立马发生 写时拷贝,子进程就有自己的缓冲区,将数据写入到缓冲区中,子进程退出后就会造成二次刷新。

  而这个现象也恰恰说明了语言层是存在缓冲区的。


✈️编码模拟

  为了更加深刻理解缓冲区这个概念,我们不妨编写一段代码来加深印象:

bash准备:

[xzy@iZ0jle4p97d8x4byf3u32mZ buffer]$ ll
total 0
-rw-rw-r-- 1 xzy xzy 0 May 13 22:54 filetest.c
-rw-rw-r-- 1 xzy xzy 0 May 13 22:54 makefile
-rw-rw-r-- 1 xzy xzy 0 May 13 22:54 mystdio.c
-rw-rw-r-- 1 xzy xzy 0 May 13 22:54 mystdio.h

mystdio.h:

#pragma once 

#include

#define SIZE 4096
#define NONE_FLUSH (1<<1)//无刷新
#define LINE_FLUSH (1<<2)//行刷新
#define FULL_FLUSH (1<<3)//全缓冲

typedef struct _myFILE
{
    char outbuffer[SIZE];//输出缓冲区
    int pos;//位置
    int cap;//容量
    int fileno;//文件描述符
    int flush_mode;//刷新方式
}myFILE;

myFILE *my_fopen(const char* pathname, const char* mode);

void my_fclose(myFILE* fp);

int my_fwrite(myFILE* fp, const char* s, int size);

mystdio.c:

#include "mystdio.h"
#include
#include
#include
#include
#include
#include

const char* toString(int flag)
{
    if(flag & NONE_FLUSH) return "None";//无缓冲
    else if(flag & LINE_FLUSH) return "Line";//flag为行缓冲
    else if(flag & FULL_FLUSH) return "FULL";//全缓冲
    return "Unknow";
}

void DebugPrint(myFILE* fp)//debug代码是否有误
{
    printf("outbuffer: %s
", fp->outbuffer);
    printf("fd: %d
", fp->fileno);
    printf("pos: %d
", fp->pos);
    printf("cap: %d
", fp->cap);
    printf("flush_mode: %s", toString(fp->flush_mode));
}

myFILE* my_fopen(const char* pathname, const char* mode)//模拟fopen函数
{
    int flag = 0;
    if(strcmp(mode, "r") == 0)//r方式打开
    {
        flag |= O_RDONLY;
    }
    else if(strcmp(mode, "w") == 0)//w 方式打开
    {
        flag |= (O_CREAT | O_WRONLY | O_TRUNC);
    }
    else if(strcmp(mode, "a") == 0)//a 方式打开
    {
        flag |= (O_CREAT | O_WRONLY | O_APPEND);
    }
    else 
    {
        return NULL;
    }

    int fd = 0;
    if(flag & O_WRONLY)//是否为写的方式打开
    {
        umask(0);
        fd = open(pathname, flag, 0666);//写的方式打开很可能会创建文件
    }
    else 
    {
        fd = open(pathname, flag);//只读方式打开
    }
    
    if(fd < 0) return NULL;

	//将FILE对象初始化
    myFILE* fp = (myFILE*)malloc(sizeof(myFILE));
    fp -> fileno = fd;
    fp -> cap = SIZE;
    fp -> pos = 0;
    fp -> flush_mode = LINE_FLUSH;//默认为行缓冲

    return fp;
}

void my_fflush(myFILE* fp)//刷新
{
    if(fp->pos == 0) return;
    write(fp->fileno, fp->outbuffer, fp->pos);
    fp->pos = 0;
}

void my_fclose(myFILE* fp)//自定义关闭文件
{
    my_fflush(fp);//退出前要刷新
    close(fp -> fileno);
    free(fp);
}

int my_fwrite(myFILE* fp, const char* s, int size)//自定义fwrite
{
    memcpy(fp->outbuffer + fp->pos, s, size);
    fp->pos += size;
    if((fp->flush_mode & LINE_FLUSH) && fp->outbuffer[fp->pos - 1] == '
') //行刷新
    {
        my_fflush(fp);   
    }
    else if((fp->flush_mode & FULL_FLUSH) && fp->pos == fp->cap)//全缓冲
    {
        my_fflush(fp);
    }
    return size;
}

filetest.c:

#include"mystdio.h"
#include

const char* filename = "./log.txt";//文件名称

int main()
{
    myFILE* fp = my_fopen(filename, "w");//以写的方式打开文件
    if(fp == NULL) return 1;

    int cnt = 5;//进行数量测试
    char buffer[64];//缓冲区
    
    while(cnt)
    {
        snprintf(buffer, sizeof(buffer), "youcanseeme, bro:%d 
", cnt--);
        my_fwrite(fp, buffer, strlen(buffer));
        sleep(2);
    }

    my_fclose(fp);
    return 0;
}

  运行成功之后,我们就可以看到现象,在log文件中打印了我们测试的内容。


📒✏️总结

  •  C语言的一些IO接口需要熟悉,例如fwrite,fputs等等。
  •  当前当前路径是根据进程的cwd来决定的,C语言默认打开三个流:stdin、stdout、stderr。他们三个 分别占用0、1、2三个文件描述符
  •  系统层面的IO交互接口有 write、open、close、read等需要理解。
  • 文件=内容+属性;一个文件是否为空都会存在属性,而操作系统为了维护文件的属性,先描述再组织,将文件的属性组织为一个结构体file,而 每个file以双链表的形式相连
  •  因为Linux下一切皆文件,所以文件也需要被组织起来,于是file结构体的指针file*被组织起来封装在一个叫做files_struct 指针数组内,而数组下标就是 文件描述符
  •  重定向是 根据更改文件描述符的指向文件 做到的,可以使用dup2接口做调整。
  •  缓冲区本质上是一块内存区域,而缓冲区分为系统层缓冲区和语言层缓冲区,在C语言中缓冲区被封装在FILE结构体内,每一个文件都有自己的缓冲区
  •  缓冲区满了会刷新到内核中,而 刷新的本质就是写入


  希望这篇文章能够帮到你【玫瑰】~~

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

搜索文章

Tags

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