最新资讯

  • 【Linux 进程控制】—— 进程亦生生不息:起于鸿蒙,守若空谷,归于太虚

【Linux 进程控制】—— 进程亦生生不息:起于鸿蒙,守若空谷,归于太虚

2025-04-25 18:59:03 1 阅读

欢迎来到一整颗红豆的博客✨,一个关于探索技术的角落,记录学习的点滴📖,分享实用的技巧🛠️,偶尔还有一些奇思妙想💡
本文由一整颗红豆原创✍️,感谢支持❤️!请尊重原创📩!欢迎评论区留言交流🌟
个人主页 👉 一整颗红豆
本文专栏➡️Linux驾驭之道 掌控操作系统的艺术与哲学

生生不息:起于鸿蒙,守若空谷,归于太虚

  • 进程创建
    • 再识fork函数
    • fork函数返回值
    • 写时拷贝 `Copy-On-Write`
  • 进程等待
    • 进程等待的方法
      • `wait`
      • `waitpid`
    • 获取子进程 status
    • 阻塞等待与非阻塞等待
      • 阻塞等待(`Blocking Wait`)
      • 非阻塞等待(`Non-blocking Wait`)
  • 进程终止
    • 进程退出场景
    • 进程退出码
    • 错误码
    • _exit函数和exit函数
    • return 退出
    • `_exit`、`exit` 和 `return` 对比
  • 写在最后

进程创建

再识fork函数

之前在 Linux进程状态 这篇文章中,我们已经为大家介绍过Linux系统中一个非常重要的系统调用 — fork,今天我们在来重谈 fork 函数,让大家对这个系统调用有更深刻的理解。

在 Linux中 fork 函数是非常重要的函数,它从已存在进程中创建⼀个新进程。创建出来的新进程叫做子进程,而原进程则称为父进程。

在Linux参考手册中,fork函数的原型如下:(man 2 fork 指令查看)

NAME
       fork - create a child process

SYNOPSIS
       #include 
       #include 

       pid_t fork(void);

如上不难看出:

  • fork 函数的功能是创建一个子进程
  • 头文件有
  • 参数为 void ,返回值为 pid_t (实际上是Linux内核中typedef出来的一个类型)

进程调用 fork,当控制转移到内核中的 fork 代码后,内核做如下几件事:

  • 分配新的内存块和内核数据结构给子进程
  • 将父进程部分数据结构内容拷贝至子进程
  • 添加子进程到系统进程列表当中
  • fork返回,开始调度器调度

当⼀个进程调用fork之后,就有两个⼆进制代码相同的进程。并且它们都运行到相同的地方。但每个进程都将可以开始属于它们自己的旅程,看如下程序:

int main(void)
{
    pid_t pid;
    printf("Before: pid is %d
", getpid());
    if ((pid = fork()) == -1)
        perror("fork()"), exit(1);
    printf("After:pid is %d, fork return %d
", getpid(), pid);
    sleep(1);
    return 0;
}

输出:

Before: pid is 40176
After:pid is 40176, fork return 40177
After:pid is 40177, fork return 0

这里看到了三行输出,⼀行before,两行after。其中 40176就是父进程啦,40177就是子进程。进程40176先打印before消息,然后它有打印after。另⼀个after消息是进程40177打印的。注意到进程40177没有打印before,为什么呢?

如下图所示:

当父进程执行到fork创建出子进程时,已经执行了上面的before代码,而创建出子进程后,子进程不会去执行父进程已经执行过的代码,而是和父进程一同执行fork之后的代码。这就是为什么子进程没有打印before的原因

所以,fork之前父进程独立执行,fork之后,父子进程两个执行流分别执行之后的代码。值得注意的是,fork之后,谁先执行完全由调度器决定,并没有明确的先后关系!


fork函数返回值

类型定义:fork() 返回 pid_t 类型(通常为 int 通过 typedef 定义),用于表示进程ID(PID)。

fork创建成功:

  • 子进程返回0
  • 父进程返回的是子进程的 pid

为什么给父进程返回子进程的pid,这个问题我们之前已经讨论过:

一个父进程可以创建一个或者多个子进程,父进程需要通过返回值获得新创建的子进程的唯一标识符(正整数),从而可以管理创建的多个子进程(如发送信号、等待终止等)

为什么子进程返回0

子进程返回0,标识自己为子进程,子进程通过返回值 0 确认自己的身份。子进程无需知晓父进程的PID(实际上可以通过 getppid() 获取)


fork创建失败:

返回 -1并设置错误码:

  • 当系统资源不足(如进程数超限、内存耗尽)时,fork() 失败。

错误码:

  • 需检查 errno 确定具体原因
if (pid == -1) {
    perror("fork failed"); // 输出类似 "fork failed: Resource temporarily unavailable"
}

常见错误码:

  • EAGAIN:进程数超过限制(RLIMIT_NPROC)或内存不足。
  • ENOMEM:内核无法分配必要数据结构所需内存。

写时拷贝 Copy-On-Write

写时拷贝(COW)是 Linux 中 fork() 系统调用的核心优化机制,它使得进程创建变得高效且资源友好,通常,父子代码共享,父子再不写入时,数据也是共享的,当任意一方试图写入,便以写时拷贝的方式各自⼀份副本。

为什么需要写时拷贝?

在传统的进程创建方式中,fork() 会直接复制父进程的所有内存空间给子进程。这种方式存在明显问题:

  • 内存浪费:如果父进程占用 1GB 内存,子进程即使不修改任何数据,也会立即消耗额外 1GB 内存。
  • 性能低下:复制大量内存需要时间,尤其是对大型进程而言,fork() 会显著延迟程序运行。

COW 的解决思路:

  • 推迟实际的内存复制,直到父子进程中某一方尝试修改内存页时,才进行真正的拷贝。在此之前,父子进程共享同一份物理内存。

具体见下图:

因为有写时拷贝技术的存在,所以父子进程得以彻底分离!完成了进程独立性的技术保证! 写时拷贝,是⼀种延时申请技术,可以提高整机内存的使用率。

写时拷贝的工作流程

1、 fork() 调用时

  • 共享内存页:内核仅为子进程创建虚拟内存结构(页表),但物理内存页仍与父进程共享。
  • 标记为只读:内核将共享的物理内存页标记为只读(即使父进程原本可写)。

2、进程尝试写入内存

  • 触发页错误:当父进程或子进程尝试修改某个共享内存页时,由于页被标记为只读,CPU 会触发页错误(Page Fault)。

内核介入处理:操作系统会由用户态陷入内核态处理异常

  • 分配新的物理内存页,复制原页内容到新页。
  • 修改触发写入的进程的页表,使其指向新页。
  • 将新页标记为可写,恢复进程执行。

3、后续操作

  • 修改后的进程独享新内存页,另一进程仍使用原页。
  • 未修改的内存页继续共享,不做复制,操作系统不做任何无意义的事情。

进程等待

之前我们在讲进程概念的时候讲过,如果父进程创建出子进程后,如果子进程已经退出,父进程却没有对子进程回收,那么就子进程就会变成 “僵尸进程” ,造成内存泄露等问题。

在Linux系统中,进程等待是父进程通过系统调用等待子进程终止并获取其退出状态的过程,主要目的是避免僵尸进程并回收子进程资源。

进程等待的必要性

僵尸进程问题:

  • 子进程终止后,其退出状态会保留在进程表中,直到父进程读取。若父进程未处理,子进程将保持僵尸状态(Zombie),占用系统资源。
  • 状态收集:父进程需知晓子进程的执行结果(成功、错误代码、信号终止等)。
  • 资源回收:内核释放子进程占用的内存、文件描述符等资源。

进程等待的方法

wait

#include
#include
pid_t wait(int* status);

具体功能:

  • 阻塞父进程,直到等待到任意一个子进程终止。

参数:

  • status:输出型参数,用来存储子进程退出状态的指针(可为 NULL,表示不关心状态)。

返回值:

  • 成功:返回终止的子进程PID。失败:返回-1(如无子进程)。

waitpid

#include 
pid_t waitpid(pid_t pid, int *status, int options);
  • 功能:更灵活的等待方式,可指定子进程或非阻塞等待模式。

参数:

pid:

  • >0:等待指定 PID 的子进程。
  • -1:等待任意子进程(等效于 wait)。
  • 0:等待同一进程组的子进程。

status:同 wait,输出型参数,表明子进程的退出状态。

options 默认为0,表示阻塞等待

  • WNOHANG:非阻塞模式,无子进程终止时立即返回 0。
  • WUNTRACED:报告已停止的子进程(如被信号暂停)。

返回值:

  • 成功:返回子进程PID。
  • WNOHANG 且无子进程终止:返回0。
  • 失败:返回-1。

做个总结:

  • 如果子进程已经退出,调用 wait / waitpid 时,wait / waitpid 会立即返回,并且释放资源,获得子进程退出信息。
  • 如果在任意时刻调用 wait / waitpid,子进程存在且正常运行,则进程可能阻塞。 如果不存在该子进程,则立即出错返回。


获取子进程 status

waitwaitpid,都有⼀个 status 参数,该参数是⼀个输出型参数,由操作系统填充。

  • 如果传递 NULL,表示不关心子进程的退出状态信息。
  • 否则,操作系统会根据该参数,将子进程的退出信息反馈给父进程。

status 不能简单的当作整形来看待,可以当作位图来看待,具体细节如下图
(只研究 status 低16比特位):


如何理解呢?
子进程的退出分为两种情况:

  1. 正常终止

高 8 位(第 8 ~ 15 位):保存子进程的退出状态(退出码)(即 exit(code)return code 中的 code 值)。

第 7 位:通常为 0,表示正常终止。

示例:

若子进程调用 exit(5),表明子进程是正常退出,则 status 的高 8 位为 00000101(即十进制 5)。

  1. 被信号所杀导致终止

低 7 位(第 0 ~ 6 位):保存导致子进程终止的信号编号。

第 7 位:若为 1,表示子进程在终止时生成了 core dump 文件(用于调试)。有关 core dump 文件,后面会讲,大家这里先了解一下即可。

第 8 ~ 15 位:未使用(通常为 0)。

示例:

若子进程因 SIGKILL(信号编号 9)终止,则 status 的低 7 位为 0001001(即十进制 9)。

  • 做个小总结:
16 位结构:
| 15 14 13 12 11 10 9 8 | 7 | 6 5 4 3 2 1 0 |
---------------------------------------------
正常终止 → [ 退出状态(高8位) ]  0  [ 未使用 ]
被信号终止 → [   未使用(全0)   ] c  [ 信号编号 ]

如何解析 status?

难道真的需要我们将 status 当作位图,使用位操作来提取子进程的退出信息吗?
这么做对我们程序员来说当然小菜一碟,不过有点多余了,没必要。Linux系统为我们定义了多种宏用来提取 status,方便且专业。

使用宏定义检查 status 的值:

功能
WIFEXITED(status)若子进程正常终止(exit 或 return)返回真。
WEXITSTATUS(status)若 WIFEXITED 为真,返回子进程的退出码(exit 的参数或 return 的值)。
WIFSIGNALED(status)若子进程因信号终止返回真。
WTERMSIG(status)若 WIFSIGNALED 为真,返回导致终止的信号编号。
WCOREDUMP(status)若子进程生成了核心转储文件返回真。

常用的两个宏:

  • WIFEXITED(status): 若为正常终止子进程返回的状态,则为真。(查看进程是 否是正常退出)
  • WEXITSTATUS(status): 若WIFEXITED非零,提取子进程退出码。(查看进程的 退出码)

示例一:子进程正常退出

int main()
{
    pid_t pid = fork();
    if (pid == 0)
    { // 子进程
        printf("子进程运行中... PID=%d
", getpid());
        // 1. 正常退出:调用 exit(42)
        exit(42);
    }
    else
    { // 父进程
        int status;
        waitpid(pid, &status, 0); // 等待子进程结束
        if (WIFEXITED(status))
        { // 正常退出
            printf("子进程正常退出,退出码: %d
", WEXITSTATUS(status));
        }
        else if (WIFSIGNALED(status))
        { // 被信号终止
            printf("子进程被信号终止,信号编号: %d
", WTERMSIG(status));
        }
    }
    return 0;
}

输出:

子进程运行中... PID=56153
子进程正常退出,退出码: 42

示例二:子进程被信号终止

int main()
{
    pid_t pid = fork();
    if (pid == 0)
    { // 子进程
        printf("子进程运行中... PID=%d
", getpid());
        int *p = NULL;
        *p = 100;  // 对空指针解引用,触发 SIGSEGV 被信号终止
    }
    else
    { // 父进程
        int status;
        waitpid(pid, &status, 0); // 等待子进程结束
        if (WIFEXITED(status))
        { // 正常退出
            printf("子进程正常退出,退出码: %d
", WEXITSTATUS(status));
        }
        else if (WIFSIGNALED(status))
        { // 被信号终止
            printf("子进程被信号终止,信号编号: %d
", WTERMSIG(status));
        }
    }
    return 0;
}

输出:

子进程运行中... PID=56203
子进程被信号终止,信号编号: 11

阻塞等待与非阻塞等待

在 Unix/Linux 中,父进程通过 wait 或 waitpid 函数等待子进程结束。它们的核心区别在于是否允许父进程在等待子进程时继续执行其他任务。

阻塞等待(Blocking Wait

父进程调用 waitpid 后,会一直挂起(阻塞),直到目标子进程终止。在阻塞期间,父进程无法执行其他操作,直到子进程退出。

pid_t waitpid(pid_t pid, int *status, 0);  // options 参数为 0

示例:

int main()
{
    int status;
    pid_t child_pid = fork();
    if (child_pid == 0)
    {
        // 子进程执行任务
        exit(10);
    }
    else
    {
        // 父进程阻塞等待子进程结束
        waitpid(child_pid, &status, 0);
        if (WIFEXITED(status))
        {
            printf("子进程退出码: %d
", WEXITSTATUS(status));
        }
    }
}

非阻塞等待(Non-blocking Wait

父进程调用 waitpid 时,若子进程未结束,则父进程立即返回,而不是挂起。父进程可以继续执行其他任务,同时定期检查子进程状态。需结合循环实现非阻塞式轮询(polling)。

关键选项:宏 WNOHANG(定义在 中)。

pid_t waitpid(pid_t pid, int *status, WNOHANG);

示例:非阻塞轮询方式

int main()
{
    int status;
    pid_t child_pid = fork();
    if (child_pid == 0)
    {
        sleep(3); // 子进程休眠 3 秒后退出
        exit(10);
    }
    else
    {
        while (1)
        {
            pid_t ret = waitpid(child_pid, &status, WNOHANG);
            if (ret == -1)
            {
                perror("waitpid");
                break;
            }
            else if (ret == 0)
            {
                printf("子进程未结束,父进程继续工作...
");
                sleep(1); // 避免频繁轮询消耗 CPU
            }
            else
            {
                if (WIFEXITED(status))
                {
                    printf("子进程退出码: %d
", WEXITSTATUS(status));
                }
                break;
            }
        }
    }
}

阻塞等待和非阻塞等待的对比:

场景阻塞等待非阻塞等待
父进程任务优先级必须立即处理子进程结果需同时处理其他任务
子进程执行时间较短或确定较长或不确定
资源消耗CPU 空闲,无额外开销需轮询,可能占用更多 CPU
典型应用简单脚本、单任务场景多进程管理、事件驱动程序

进程终止

进程= 内核数据结构 + 进程自己的代码和数据

进程终止是进程生命周期的最后一个阶段,涉及资源释放、状态通知及父进程回收等关键步骤。进程终止的本质是释放系统资源,就是释放进程申请的相关内核数据结构和对应的代码和数据。

进程退出场景

  • 代码运行完毕,结果正确
  • 代码运行完毕,结果不正确
  • 代码异常终止

如何理解这三种进程退出的场景呢?举个例子

代码运行完毕,结果正确

  • 程序完整执行了所有逻辑,未触发任何错误或异常。
  • 输出结果与预期完全一致,符合功能需求或算法目标。
int sum(int a, int b)
{
    return a + b;
}

int main()
{
    int result = sum(3, 5);
    printf("Result: %d
", result); // 输出 8,结果正确
    return 0;
}

输出:

Result: 8

代码运行完毕,结果不正确

  • 程序正常结束(无崩溃或异常),但输出结果与预期不符。
  • 通常由逻辑错误、算法错误或数据处理错误导致。

例如:

// 错误实现:本应计算阶乘,但初始值错误
int factorial(int n)
{
    int result = 0; // 错误!应为 result = 1
    for (int i = 1; i <= n; i++)
    {
        result *= i;
    }
    return result;
}

int main()
{
    printf("5! = %d
", factorial(5)); // 输出 0,结果错误
    return 0;
}

代码未执行完毕,异常终止

  • 程序未执行完毕就中途崩溃或被强制终止。
  • 通常由运行时错误、资源限制或外部信号触发。
  • 比如除零错误,对空指针解引用等异常

例如

int main()
{
    int *ptr = NULL;
    *ptr = 42;  // 对空指针解引用,触发段错误
    printf("Value: %d
", *ptr);
    return 0;
}

段错误:

Segmentation fault

再比如:

int main()
{
    int a = 10;
    int b = a / 0; // 程序除零异常
    printf("Value: %d
", b);
    return 0;
}

浮点数异常:

Floating point exception

进程常见退出方法

正常终止(可以通过 echo $? 查看进程退出码)

  1. 从main返回
  2. 调用exit
  3. _exit

异常退出:

  1. ctrl + c,信号终止

进程退出码

进程退出码(退出状态)可以告诉我们最后⼀次执行的命令的状态。在命令结束以后,我们可以知道命令是成功完成的还是以错误结束的。通常是你程序中mian函数的返回值,其基本思想是,程序返回退出代码 0 时表示执行成功,没有问题。 0 以外的任何代码都被视为不成功。

退出码是一个 8 位无符号整数(8-bit unsigned integer),因此取值范围为 2^8=256 个值。

Linux Shell 中的常见退出码:

  • 退出码 0 表示命令执行有误,这是完成命令的理想状态。
  • 退出码 1 我们也可以将其解释为 “不被允许的操作”。例如在没有 sudo 权限的情况下使用 yum
  • 130 ( SIGINT 或 ^C )和 143 ( SIGTERM )等终止信号是非常典型的,它们属于 128+n 信号,其中 n 代表信号编号。

这里需要补充一点:

进程退出码和错误码是两个完全不同的概念,不要混为一谈!

错误码

在 Linux 系统中,错误码(Error Codes)是操作系统用于标识程序运行中遇到的各类问题的核心机制。这些错误码通过全局变量 errno(定义在 头文件中)传递,帮助开发者快速定位和调试问题。

要理解错误码,首先要认识全局变量 error

例如:fork函数调用失败后,会立刻返回-1,并设置全局变量 error

  • 定义:errno 是一个线程安全的整型变量,用于存储最近一次系统调用或库函数调用失败的错误码。

特性:

  • 成功调用不会重置 errno,因此必须在调用后立即检查其值。
  • 每个线程有独立的 errno 副本(多线程安全)。

头文件:

#include 

与之对应的是 strerror 函数,该函数可以将对应的错误码转化成字符串描述的错误信息打印出来,方便程序员调试代码。

实际上,我们可以通过 for 循环来打印查看Linux系统下所有的错误码以及其错误信息:

int main()
{
    for (int i = 0; i < 135; ++i)
    {
        printf("%d-> %s
", i, strerror(i));
    }
    return 0;
}

不难看出,在Linux系统下,一共有 0 ~ 133 总共134个错误码,其中 0 表示 success ,即程序运行成功, 1 ~ 133 则分别对应一个错误信息。

0-> Success
1-> Operation not permitted
2-> No such file or directory
3-> No such process
4-> Interrupted system call
5-> Input/output error
6-> No such device or address
7-> Argument list too long
8-> Exec format error
9-> Bad file descriptor
10-> No child processes
11-> Resource temporarily unavailable
12-> Cannot allocate memory
13-> Permission denied
14-> Bad address
15-> Block device required
16-> Device or resource busy
17-> File exists
18-> Invalid cross-device link
19-> No such device
20-> Not a directory
21-> Is a directory
22-> Invalid argument
23-> Too many open files in system
24-> Too many open files
25-> Inappropriate ioctl for device
26-> Text file busy
27-> File too large
28-> No space left on device
29-> Illegal seek
30-> Read-only file system
31-> Too many links
32-> Broken pipe
33-> Numerical argument out of domain
34-> Numerical result out of range
35-> Resource deadlock avoided
36-> File name too long
37-> No locks available
38-> Function not implemented
39-> Directory not empty
40-> Too many levels of symbolic links
41-> Unknown error 41
42-> No message of desired type
43-> Identifier removed
44-> Channel number out of range
45-> Level 2 not synchronized
46-> Level 3 halted
47-> Level 3 reset
48-> Link number out of range
49-> Protocol driver not attached
50-> No CSI structure available
51-> Level 2 halted
52-> Invalid exchange
53-> Invalid request descriptor
54-> Exchange full
55-> No anode
56-> Invalid request code
57-> Invalid slot
58-> Unknown error 58
59-> Bad font file format
60-> Device not a stream
61-> No data available
62-> Timer expired
63-> Out of streams resources
64-> Machine is not on the network
65-> Package not installed
66-> Object is remote
67-> Link has been severed
68-> Advertise error
69-> Srmount error
70-> Communication error on send
71-> Protocol error
72-> Multihop attempted
73-> RFS specific error
74-> Bad message
75-> Value too large for defined data type
76-> Name not unique on network
77-> File descriptor in bad state
78-> Remote address changed
79-> Can not access a needed shared library
80-> Accessing a corrupted shared library
81-> .lib section in a.out corrupted
82-> Attempting to link in too many shared libraries
83-> Cannot exec a shared library directly
84-> Invalid or incomplete multibyte or wide character
85-> Interrupted system call should be restarted
86-> Streams pipe error
87-> Too many users
88-> Socket operation on non-socket
89-> Destination address required
90-> Message too long
91-> Protocol wrong type for socket
92-> Protocol not available
93-> Protocol not supported
94-> Socket type not supported
95-> Operation not supported
96-> Protocol family not supported
97-> Address family not supported by protocol
98-> Address already in use
99-> Cannot assign requested address
100-> Network is down
101-> Network is unreachable
102-> Network dropped connection on reset
103-> Software caused connection abort
104-> Connection reset by peer
105-> No buffer space available
106-> Transport endpoint is already connected
107-> Transport endpoint is not connected
108-> Cannot send after transport endpoint shutdown
109-> Too many references: cannot splice
110-> Connection timed out
111-> Connection refused
112-> Host is down
113-> No route to host
114-> Operation already in progress
115-> Operation now in progress
116-> Stale file handle
117-> Structure needs cleaning
118-> Not a XENIX named type file
119-> No XENIX semaphores available
120-> Is a named type file
121-> Remote I/O error
122-> Disk quota exceeded
123-> No medium found
124-> Wrong medium type
125-> Operation canceled
126-> Required key not available
127-> Key has expired
128-> Key has been revoked
129-> Key was rejected by service
130-> Owner died
131-> State not recoverable
132-> Operation not possible due to RF-kill
133-> Memory page has hardware error
134-> Unknown error 134

错误码的应用:

int main()
{
    FILE *fp = fopen("invalid.txt", "r");//以只读方式打开不存在的文件会出错
    if (fp == NULL)
    {
        // 使用 strerror 获取错误描述
        printf("%d->%s
", errno,strerror(errno));            
        return 1; //退出码设为1
    }
    return 0;
}

输出:

2->No such file or directory

使用错误码和对应的错误信息可以帮助程序员快速定位错误模块,调试程序,掌握错误码的使用与调试技巧,是提升 Linux 编程效率和系统可靠性的关键。


_exit函数和exit函数

_exit函数

在 Linux 系统中,_exit() 是一个直接终止进程的系统调用,它会立即终止当前进程,并通知操作系统回收资源,但不执行任何用户空间的清理操作。

#include 
void _exit(int status);
  • 参数 status:进程的退出状态码,范围是 0~255。父进程可以通过 wait()waitpid() 获取该状态码。
  • 返回值:无(进程直接终止,不会返回调用者)。

当前进程调用 _exit() 后,操作系统会立即介入,会从用户态陷入内核态,执行以下操作:

  • 关闭所有文件描述符:内核会关闭进程打开的文件、套接字、管道等资源,但不会刷新标准 I/O 库(如 stdio)的缓冲区。
  • 释放用户空间内存:回收进程的代码段、数据段、堆、栈等内存资源。
  • 发送 SIGCHLD 信号: 通知父进程子进程已终止,并传递退出状态码 status
  • 终止进程:进程的状态变为 ZOMBIE(僵尸进程),直到父进程通过 wait() 回收其资源。

本质上,_exit() 最终会调用 Linux 内核的 exit_group 系统调用(sys_exit_group),终止整个进程及其所有线程。其内核处理流程如下:

释放进程资源:

  • 关闭所有文件描述符。
  • 释放内存映射(mmap)和虚拟内存区域。
  • 解除信号处理程序绑定。

更新进程状态:

  • 将进程状态设为 TASK_DEAD
  • 向父进程发送 SIGCHLD 信号。

调度器介入:

  • 从运行队列中移除进程。
  • 切换到下一个进程执行。

exit函数

在 C/C++ 语言中,exit 是一个用于正常终止程序执行的标准库函数。它会执行一系列清理操作后终止进程,并将控制权交还给操作系统。

#include 
void exit(int status);  // C 

#include 
void exit(int status);  // C++ 
  • 参数 status:进程的退出状态码,范围 0~255(0 通常表示成功,非零表示异常)。
  • 返回值:无(进程终止,不会返回调用者)。

进程调用 exit 时,按以下顺序执行操作:

  1. 调用 atexit 注册的函数:按注册的逆序执行所有通过 atexit
    at_quick_exit(若使用quick_exit)注册的函数。
  2. 刷新所有标准 I/O 缓冲区:清空 stdoutstderr 等流的缓冲区。 注意: stderr 默认无缓冲,stdout 在交互式设备上是行缓冲。
  3. 关闭所有打开的文件流:调用 fclose 关闭所有通过 fopen 打开的文件。 注意:不会关闭底层文件描述符(需手动 close)。
  4. 删除临时文件:删除由 tmpfile 创建的临时文件。
  5. 终止进程:向操作系统返回状态码 status。父进程可通过 waitwaitpid 获取该状态码。

其实本质上,exit 是一个标准库函数,最后也会调用_exit,但是在这之前,exit还做了其他的清理工作:

我们举个例子,帮大家直观的感受一下这两者的区别:

示例一:使用 exit 函数

int main()
{
    printf("hello");
    exit(0);
}

输出:

[root@localhost linux]# ./a.out
hello[root@localhost linux]#

示例二:使用 _exit 函数

int main()
{
    printf("hello");
    _exit(0);
}

输出:

[root@localhost linux]# ./a.out
[root@localhost linux]#

聪明的同学很快就知道了,我们通过 printf 打印 “hello” 并没有加上换行符,所以“hello”
在缓冲区内没有被立即刷新,所以当我们使用exit终止进程时,exit会帮我们做相应的清理工作,包括刷新I/O缓冲区。而调用_exit时则不会刷新,进程直接退出。

return 退出

return是⼀种更常见的进程退出方法。执行 return n 等同于执行 exit(n),因为调用main的运行时函数会将main函数的返回值当做 exit 的参数。

状态码传递:

main函数中的 return 语句返回一个整数值(通常称为退出状态码),表示程序的执行结果:

  • 0:表示程序成功执行。
  • 非0:表示程序异常终止(具体数值由程序员定义)。

return与exit()的关系

隐式调用exit():

  • 在 main 函数中使用 return 时,C/C++运行时会自动调用 exit() 函数,并将返回值作为参数传递给它。
int main()
{
    return 42;  // 等价于 exit(42);
}

return的执行流程

当在main函数中执行return时,程序会做以下几件事:

  • 返回值传递:将返回值传递给运行时环境。

清理操作:

  • 调用局部对象的析构函数(按照创建顺序的逆序)。
  • 调用全局对象的析构函数(同样逆序)。

调用exit():运行时调用exit(),执行以下操作:

  • 刷新所有I/O缓冲区(如 std::cout)。
  • 关闭通过 fopen 打开的文件流。
  • 执行通过 atexit() 注册的函数。

终止进程:将控制权交还给操作系统。

值得注意的一点是:在非main函数的其他函数中使用 return 仅退出当前函数,返回到调用者,不会终止进程。


_exitexitreturn 对比

以下是一个详细的表格供大家理解参考

特性_exit() (系统调用)exit() (标准库函数)return (在 main 中)
所属标准POSIX 系统调用C/C++ 标准库函数C/C++ 语言关键字
头文件(C)、(C++)无(语言内置)
执行流程立即终止进程,不执行任何用户空间清理。1. 调用 atexit 注册的函数
2. 刷新 I/O 缓冲区
3. 关闭文件流
1. 调用 C++ 局部对象析构函数
2. 隐式调用 exit() 完成后续清理
清理操作内核自动回收进程资源(内存、文件描述符),不刷新缓冲区、不调用析构函数清理标准库资源(刷新缓冲区、关闭文件流),但不调用 C++ 局部对象析构函数调用 C++ 局部和全局对象析构函数,并触发 exit() 的清理逻辑
多线程行为立即终止所有线程,可能导致资源泄漏终止整个进程,但可能跳过部分线程资源释放(如线程局部存储)同 exit(),但在 C++ 中会正确析构主线程的局部对象
C++ 析构函数调用❌ 不调用任何对象的析构函数(包括全局对象)❌ 不调用局部对象析构函数
✅ 调用全局对象析构函数(C++)
✅ 调用局部和全局对象析构函数(C++)
缓冲区处理❌ 不刷新 stdio 缓冲区(如 printf 的输出可能丢失)✅ 刷新所有 stdio 缓冲区✅ 通过隐式调用 exit() 刷新缓冲区
适用场景1. 子进程退出(避免重复刷新缓冲区)
2. 需要立即终止进程(绕过清理逻辑)
1. 非 main 函数的程序终止
2. 需要执行注册的清理函数(如日志收尾)
1. 在 main 函数中正常退出
2. 需要确保 C++ 对象析构(RAII 资源管理)
错误处理直接传递状态码给操作系统,无错误反馈机制可通过 atexit 注册错误处理函数,但无法捕获局部对象析构异常可通过 C++ 异常机制处理错误(需在 main 中捕获)
信号安全✅ 可在信号处理函数中安全调用(如 SIGINT)❌ 不可在信号处理函数中调用(可能死锁)❌ 不可在信号处理函数中使用(仅限 main 函数流程)
资源泄漏风险高(临时文件、未释放的手动内存等需内核回收)中(未关闭的文件描述符、手动内存需提前处理)低(依赖 RAII 自动释放资源)
底层实现直接调用内核的 exit_group 系统调用调用 C 标准库的清理逻辑后,最终调用 _exit()编译器生成代码调用析构函数,并跳转到 main 结尾触发 exit()

最后总结下:

  • _exit():最底层的终止方式,适合需要绕过所有用户空间清理的场景(如子进程退出)。
  • exit():平衡安全与效率,适合非 main 函数的程序终止,但需注意 C++ 对象析构问题。
  • return:C++ 中最安全的退出方式,优先在 main 函数中使用,确保资源自动释放。

写在最后

本文到这里就结束了,后面的文章我们会展开讲解有关 Linux操作系统的更多话题,带你从新手小白 成长为一名 Linux 糕手, 感谢您的观看!

如果你觉得这篇文章对你有所帮助,请为我的博客 点赞👍收藏⭐️ 评论💬或 分享🔗 支持一下!你的每一个支持都是我继续创作的动力✨!🙏
如果你有任何问题或想法,也欢迎 留言💬 交流,一起进步📚!❤️ 感谢你的阅读和支持🌟!🎉
祝各位大佬吃得饱🍖,睡得好🛌,日有所得📈,逐梦扬帆⛵!

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

搜索文章

Tags

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