最新资讯

  • Linux-基础IO

Linux-基础IO

2025-05-08 21:01:05 0 阅读

🌎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/9284.html

搜索文章

Tags

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