【Linux】进程控制&实现自主shell

一、进程创建1、fork函数初识

linux中fork函数是非常重要的函数,它从已存在进程中创建一个新进程。新进程为子进程,而原进程为父进程。

代码语言:javascript代码运行次数:0运行复制

#include pid_t fork(void);返回值:⾃进程中返回0,⽗进程返回⼦进程id,出错返回-1

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

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

【Linux】进程控制&实现自主shell

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

代码语言:javascript代码运行次数:0运行复制

int main(){pid_t pid;printf("Before: pid is %dn", getpid());if ( (pid=fork()) == -1 ){perror("fork()");exit(1);}printf("After:pid is %d, fork return %dn", getpid(), pid);sleep(1);return 0;} **运⾏结果**:[root@localhost linux]# ./a.outBefore: pid is 43676After:pid is 43676, fork return 43677After:pid is 43677, fork return 0

这里看到了三行输出,一行before,两行after。进程43676先打印before消息,然后它有打印after。另一个after消息有43677打印的。注意到进程43677没有打印before,为什么呢?如下图所示

【Linux】进程控制&实现自主shell

所以,fork之前父进程独立执行,fork之后,父子两个执行流分别执行。注意,fork之后,谁先执行完全由调度器决定。

2、fork函数返回值子进程返回0,父进程返回的是子进程的pid。3、写时拷贝

通常,父子代码共享,父子在不写入时,数据也是共享的,当任意一方试图写入,便以写时拷贝的方式各自一份副本。具体见下图:

【Linux】进程控制&实现自主shell

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

4、fork常规用法一个父进程希望复制自己,使父子进程同时执行不同的代码段。例如,父进程等待客户端请求,生成子进程来处理请求。一个进程要执行一个不同的程序。例如子进程从fork返回后,调用exec函数。5、fork调用失败的原因系统中有太多的进程实际用户的进程数超过了限制二、进程终止

进程终止的本质是释放系统资源,就是释放进程申请的相关内核数据结构和对应的数据和代码。

1、进程退出场景代码运行完毕,结果正确代码运行完毕,结果不正确代码异常终止2、进程常见退出方法

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

从main返回调用exit_exit

异常退出: • ctrl + c,信号终止

2.1、退出码

退出码(退出状态)可以告诉我们最后一次执行的命令的状态。在命令结束以后,我们可以知道命令是成功完成的还是以错误结束的。其基本思想是,程序返回退出代码 0 时表示执行成功,没有问题。代码 1 或 0 以外的任何代码都被视为不成功。 Linux Shell 中的主要退出码:

【Linux】进程控制&实现自主shell

退出码 0 表示命令执行无误,这是完成命令的理想状态。退出码 1 我们也可以将其解释为 “不被允许的操作”。例如在没有 sudo 权限的情况下使用apt;再例如除以 0 等操作也会返回错误码 1 ,对应的命令为 let a=1/0130SIGINT^C )和 143SIGTERM )等终止信号是非常典型的,它们属于 128+n 信号,其中 n 代表终止码。可以使用strerror函数来获取退出码对应的描述。2.2、_exit函数代码语言:javascript代码运行次数:0运行复制

#include void _exit(int status);参数:status 定义了进程的终⽌状态,⽗进程通过wait来获取该值

说明:虽然status是int,但是仅有低8位可以被父进程所用。所以_exit(-1)时,在终端执行$?发现返回值是255。2.3、exit函数代码语言:javascript代码运行次数:0运行复制

#include void exit(int status);

exit最后也会调用_exit, 但在调用_exit之前,还做了其他工作:

执行用户通过 atexit或on_exit定义的清理函数。关闭所有打开的流,所有的缓存数据均被写入调用_exit

【Linux】进程控制&实现自主shell

实例:

代码语言:javascript代码运行次数:0运行复制

int main(){printf("hello");exit(0);} **运⾏结果**:[root@localhost linux]# ./a.outhello[root@localhost linux]#//int main(){printf("hello");_exit(0);} **运⾏结果**:[root@localhost linux]# ./a.out[root@localhost linux]#

2.4、return退出

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

三、进程等待1、进程等待必要性之前讲过,子进程退出,父进程如果不管不顾,就可能造成‘僵尸进程’的问题,进而造成内存泄漏。另外,进程一旦变成僵尸状态,那就刀枪不入,“杀人不眨眼”的kill -9 也无能为力,因为谁也没有办法杀死一个已经死去的进程。最后,父进程派给子进程的任务完成的如何,我们需要知道。如,子进程运行完成,结果对还是不对,或者是否正常退出。父进程通过进程等待的方式,回收子进程资源,获取子进程退出信息2、进程等待的方法2.1、wait方法代码语言:javascript代码运行次数:0运行复制

#include#includepid_t wait(int* status);返回值:成功返回被等待进程pid,失败返回-1。参数:输出型参数,获取⼦进程退出状态,不关⼼则可以设置成为NULL

2.2、waitpid方法代码语言:javascript代码运行次数:0运行复制

pid_ t waitpid(pid_t pid, int *status, int options);返回值:当正常返回的时候waitpid返回收集到的⼦进程的进程ID;如果设置了选项WNOHANG,⽽调⽤中waitpid发现没有已退出的⼦进程可收集,则返回0;如果调⽤中出错,则返回-1,这时errno会被设置成相应的值以指⽰错误所在;参数:pid:Pid=-1,等待任⼀个⼦进程。与wait等效。Pid>0.等待其进程ID与pid相等的⼦进程。status: 输出型参数WIFEXITED(status): 若为正常终⽌⼦进程返回的状态,则为真。(查看进程是否是正常退出)WEXITSTATUS(status): 若WIFEXITED⾮零,提取⼦进程退出码。(查看进程的退出码)options:默认为0,表⽰阻塞等待WNOHANG: 若pid指定的⼦进程没有结束,则waitpid()函数返回0,不予以等待。若正常结束,则返回该⼦进程的ID。

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

【Linux】进程控制&实现自主shell

2.3、获取子进程statuswait和waitpid,都有一个status参数,该参数是一个输出型参数,由操作系统填充。如果传递NULL,表示不关心子进程的退出状态信息。否则,操作系统会根据该参数,将子进程的退出信息反馈给父进程。status不能简单的当作整形来看待,可以当作位图来看待,具体细节如下图(只研究status低16比特位):

【Linux】进程控制&实现自主shell

测试代码:

代码语言:javascript代码运行次数:0运行复制

#include #include #include #include #include int main( void ){pid_t pid;if ( (pid=fork()) == -1 )perror("fork"),exit(1);if ( pid == 0 ){sleep(20);exit(10);} else {int st;int ret = wait(&st);if ( ret > 0 && ( st & 0X7F ) == 0 ){ // 正常退出printf("child exit code:%dn", (st>>8)&0XFF);} else if( ret > 0 ) { // 异常退出printf("sig code : %dn", st&0X7F );}}} **测试结果**:# ./a.out #等20秒退出child exit code:10# ./a.out #在其他终端kill掉sig code : 9

2.4、阻塞与非阻塞等待进程的阻塞等待方式:代码语言:javascript代码运行次数:0运行复制

int main(){pid_t pid;pid = fork();if(pid < 0){printf("%s fork errorn",__FUNCTION__);return 1;} else if( pid == 0 ){ //childprintf("child is run, pid is : %dn",getpid());sleep(5);exit(257);} else{int status = 0;pid_t ret = waitpid(-1, &status, 0);//阻塞式等待,等待5Sprintf("this is test for waitn");if( WIFEXITED(status) && ret == pid ){printf("wait child 5s success, child return code is:%d.n",WEXITSTATUS(status));}else{printf("wait child failed, return.n");return 1;}} return 0;} **运⾏结果**:[root@localhost linux]# ./a.outchild is run, pid is : 45110this is test for waitwait child 5s success, child return code is :1.

进程的非阻塞等待方式:代码语言:javascript代码运行次数:0运行复制

#include #include #include #include #include typedef void (*handler_t)(); // 函数指针类型std::vector handlers; // 函数指针数组void fun_one() {printf("这是⼀个临时任务1n");} void fun_two() {printf("这是⼀个临时任务2n");} void Load() {handlers.push_back(fun_one);handlers.push_back(fun_two);} void handler() {if (handlers.empty())Load();for (auto iter : handlers)iter();} int main() {pid_t pid;pid = fork();if (pid < 0) {printf("%s fork errorn", __FUNCTION__);return 1;} else if (pid == 0) { // childprintf("child is run, pid is : %dn", getpid());sleep(5);exit(1);} else {int status = 0;pid_t ret = 0;do {ret = waitpid(-1, &status, WNOHANG); // ⾮阻塞式等待if (ret == 0) {printf("child is runningn");} handler();} while (ret == 0);if (WIFEXITED(status) && ret == pid) {printf("wait child 5s success, child return code is :%d.n",WEXITSTATUS(status));} else {printf("wait child failed, return.n");return 1;}} return 0;}

四、进程程序替换

fork() 之后,父子各自执行父进程代码的一部分如果子进程就想执行一个全新的程序呢?进程的程序替换来完成这个功能! 程序替换是通过特定的接口,加载磁盘上的一个全新的程序(代码和数据),加载到调用进程的地址空间中!

1、替换原理

用fork创建子进程后执行的是和父进程相同的程序(但有可能执行不同的代码分⽀),子进程往往要调用一种exec函数以执行另一个程序。当进程调用一种exec函数时,该进程的用户空间代码和数据完全被新程序替换,从新程序的启动例程开始执行。调用exec并不创建新进程,所以调用exec前后该进程的id并未改变。

【Linux】进程控制&实现自主shell

2、替换函数

其实有六种以exec开头的函数,统称exec函数: #include int execl(const char *path, const char *arg, …); int execlp(const char *file, const char *arg, …); int execle(const char *path, const char *arg, …,char *const envp[]); int execv(const char *path, char *const argv[]); int execvp(const char *file, char *const argv[]); int execve(const char *path, char *const argv[], char *const envp[]); 1 2 3 4 5 6 7 8

2.1、函数解释这些函数如果调用成功则加载新的程序从启动代码开始执行,不再返回。如果调用出错则返回-1所以exec函数只有出错的返回值而没有成功的返回值。2.2、命名理解

这些函数原型看起来很容易混,但只要掌握了规律就很好记。

l(list) : 表示参数采用列表v(vector) : 参数用数组p(path) : 有p自动搜索环境变量PATHe(env) : 表示自己维护环境变量

函数名

参数格式

是否带路径

是否使用当前环境变量

execl

列表

不是

ViiTor实时翻译 ViiTor实时翻译

AI实时多语言翻译专家!强大的语音识别、AR翻译功能。

ViiTor实时翻译 116 查看详情 ViiTor实时翻译

execlp

列表

execle

列表

不是

不是,须自己组装环境变量

execv

数组

不是

execvp

数组

execve

数组

不是

不是,须自己组装环境变量

exec调用举例如下:

代码语言:javascript代码运行次数:0运行复制

#include int main(){char *const argv[] = {"ps", "-ef", NULL};char *const envp[] = {"PATH=/bin:/usr/bin", "TERM=console", NULL};execl("/bin/ps", "ps", "-ef", NULL);// 带p的,可以使⽤环境变量PATH,⽆需写全路径execlp("ps", "ps", "-ef", NULL);// 带e的,需要⾃⼰组装环境变量execle("ps", "ps", "-ef", NULL, envp);execv("/bin/ps", argv);// 带p的,可以使⽤环境变量PATH,⽆需写全路径execvp("ps", argv);// 带e的,需要⾃⼰组装环境变量execve("/bin/ps", argv, envp);exit(0);}

事实上,只有execve是真正的系统调用,其它五个函数最终都调用execve,所以execve在man手册 第2节,其它函数在man手册第3节。这些函数之间的关系如下图所示。 下图exec函数簇 一个完整的例子:

【Linux】进程控制&实现自主shell

五、自主Shell命令行解释器1、目标要能处理普通命令要能处理内建命令要能帮助我们理解内建命令/本地变量/环境变量这些概念要能帮助我们理解shell的允许原理2、实现原理

考虑下面这个与shell典型的互动:

代码语言:javascript代码运行次数:0运行复制

[root@localhost epoll]# lsclient.cpp  readme.md  server.cpp  utility.h[root@localhost epoll]# ps   PID TTY     TIME CMD3451 pts/0  00:00:00 bash3514 pts/0  00:00:00 ps 1 2 3 4 5 6

用下图的时间轴来表示事件的发生次序。其中时间从左向右。shell由标识为sh的方块代表,它随着时间的流逝从左向右移动。shell从用户读入字符串”ls”。shell建立一个新的进程,然后在那个进程中运行ls程序并等待那个进程结束。

【Linux】进程控制&实现自主shell

然后shell读取新的一行输入,建立一个新的进程,在这个进程中运行程序 并等待这个进程结束。 所以要写一个shell,需要循环以下过程:

获取命令行解析命令行建立一个子进程(fork)替换子进程(execvp)父进程等待子进程退出(wait)

根据这些思路,和我们前面的学的技术,就可以自己来实现一个shell了。

3、shell源码

实现代码(点击跳转);

4、总结

思考函数和进程之间的相似性 exec/exit就像call/return

一个C程序有很多函数组成。一个函数可以调用另外一个函数,同时传递给它一些参数。被调用的函数执行一定的操作,然后返回一个值。每个函数都有他的局部变量,不同的函数通过call/return系统进行通信。

这种通过参数和返回值在拥有私有数据的函数间通信的模式是结构化程序设计的基础。Linux⿎励将这种应用于程序之内的模式扩展到程序之间。如下图

【Linux】进程控制&实现自主shell

一个C程序可以fork/exec另一个程序,并传给它一些参数。这个被调用的程序执行一定的操作,然后通过exit(n)来返回值。调用它的进程可以通过wait(&ret)来获取exit的返回值。

以上就是【Linux】进程控制&实现自主shell的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/485809.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月8日 13:21:33
下一篇 2025年11月8日 13:22:08

相关推荐

  • 为什么合约新手要避开小市值山寨?控盘严重导致的定向爆破

    小市值山寨币因流动性差、持仓集中易被控盘,新手合约交易者需警惕定向爆破风险。首先应识别高风险币种,避开日均成交额低于1000万美元、订单簿深度不足、前10地址持币超50%的项目,并排除交易所评级为“高风险”的品种。其次调整交易策略,将杠杆控制在5倍以内,单笔亏损不超过账户总额的1%,并通过止损反推仓…

    2025年12月11日
    000
  • 警惕幸存者偏差:为什么你看到的成功案例可能都是陷阱

    幸存者偏差是因只关注成功而忽略失败导致的认知错误,需识别沉默数据、验证多角度信息并建立系统评估框架以避免决策失误。 在分析成功案例时,人们常忽略失败的大多数,仅凭幸存者数据做出判断,导致决策失误。 为了方便新手快速上手币圈交易并实时查看市场数据,可通过主流交易所币安(Binance)或欧易OKX注册…

    2025年12月11日
    000
  • Curve Finance的工作原理是什么?为何它被称为稳定币兑换之王

    Curve Finance通过StableSwap算法优化稳定币兑换,实现低滑点与高资本效率;算法结合恒定和与乘积模型,在小额交易时保持1:1汇率降低滑点,大额交易时自动调整维持平衡;针对稳定币锚定特性,集中流动性于价格曲线平坦区,显著压缩无常损失,使LP在低风险下持续赚取手续费;并通过CRV与ve…

    2025年12月11日
    000
  • Cookie币staking收益研究_长期持有价值分析

    COOKIE币质押需先确认账户余额并连接兼容链,通过官方页面锁定代币获取基于通胀模型的动态收益,APY随市场波动;高收益率不保证净收益,须结合币价表现计算实际回报,避免因价格下跌导致亏损;不同锁仓周期影响流动性,长期质押收益更高但牺牲资金灵活性,需权衡机会成本;生态发展决定代币价值支撑,应关注项目进…

    2025年12月11日
    000
  • Render Token (RNDR)币应用场景_未来五年价格预测报告

    Render网络通过连接全球GPU资源实现去中心化渲染,支持AI训练、元宇宙开发与影视特效制作,任务由分布式节点并行处理,结果经区块链验证后以RNDR代币结算,已成功应用于《流浪地球》等影片的高精度渲染,大幅提升效率并降低成本。 一、Render网络的去中心化渲染原理 Render网络通过连接全球闲…

    2025年12月11日
    000
  • 什么是比特币“减半”?它为什么被视为牛市启动的关键信号

    比特币减半是每四年区块奖励减半的通缩机制,2024年4月奖励降至3.125 BTC,预计2028年进一步降至1.5625 BTC,最终在2140年挖完;因供应减少、历史上涨规律及市场关注度提升,减半常被视为牛市信号。 比特币“减半”是其协议中预设的通缩机制,大约每四年将矿工获得的区块奖励减少一半。 …

    2025年12月11日
    000
  • Optimism的“超级链”(Superchain)愿景是什么?

    Optimism的“超级链”愿景通过模块化与去中心化技术构建统一生态系统。一、实现跨链无缝交互:采用标准化通信协议,部署跨链消息传递桥接合约,支持双向通信;统一地址格式映射规则,避免用户混淆;引入轻客户端验证机制,提升交易确认速度。二、共享排序器网络:多链共用去中心化排序器,节点分布于全球运营商,通…

    2025年12月11日
    000
  • Arbitrum (ARB)币项目深度研究_长期发展路线与价格展望

    Arbitrum(ARB)是以太坊Layer 2扩容方案,基于Optimistic Rollup技术提升效率与安全,采用交互式欺诈证明、Nitro升级及AnyTrust机制优化性能与去中心化;ARB代币用于DAO治理、潜在质押收益及生态燃料支付;通过Orbit框架支持定制化应用链,推动模块化L2生态…

    2025年12月11日
    000
  • 为什么需要避免频繁查看行情?克服短线交易的诱惑

    频繁查看行情易引发情绪化交易,应通过识别波动本质、建立物理隔离、重构反馈系统和强化认知训练来改善。1、忽略短期噪音,记录交易冲动并复盘偏差;2、卸载移动端软件,屏蔽网站,设定固定查看时段;3、以策略执行合规率考核交易,建立日志与惩罚机制;4、用深度研究替代盯盘,开展项目制学习积累长期价值信息。 频繁…

    2025年12月11日
    000
  • COOKIE币流通机制分析_2029-2045年市值预测

    COOKIE币总发行1亿枚,46%流通,56.9%用于生态激励,14.76%用于质押与上市,团队持股9%并设解锁机制,预种子轮占9.35%;市值受技术迭代、生态合作、质押收益、交易活跃度及社区治理影响;历史最高价$0.6697,现价约$0.0047,市场情绪中性偏贪婪,24H换手率0.11%,成交额…

    2025年12月11日
    000
  • 什么是标记价格与最新成交价?为什么K线没跌到却爆仓了

    标记价格是交易所防操纵的参考价,用于计算盈亏和强平,最新成交价是实际交易价格构成K线;当标记价格跌破强平价时即触发爆仓,即使K线显示的最新价未达止损位,主因是两者差异及高杠杆放大风险。 binance币安交易所 注册入口: APP下载: 欧易OKX交易所 注册入口: APP下载: 火币交易所: 注册…

    2025年12月11日
    000
  • “Gas空投”是什么?为什么它是最公平的空投方式之一?

    Gas空投通过报销用户交互中的Gas费用来降低参与门槛,精准奖励真实行为。1、项目方利用链上数据识别完成特定操作的地址;2、按实际消耗Gas折算补偿金额,以稳定币或代币发放;3、设定交互次数、功能覆盖等标准,结合分级报销与时间权重,防止巨鲸垄断,提升去中心化与公平性。 “Gas空投”指项目方补偿用户…

    2025年12月11日
    000
  • 什么是合约最大杠杆限制?为什么新账户不能开125倍杠杆

    合约最大杠杆限制是交易所设定的单仓位最高借贷倍数,如125倍即1单位本金可控制125单位合约,能放大收益与风险。不同平台依合约品种设不同上限,并动态调整。新账户常无法直接开通125倍杠杆,主因包括未完成高级认证、无交易记录被归为保守型、存在风险事件或所选合约不支持。用户可通过提升认证等级、增加交易频…

    2025年12月11日
    000
  • 拒绝由于信息差亏钱,这5个币圈必备网站请收藏!

    CoinMarketCap提供全球加密货币数据,支持价格监控与资讯获取;2. CoinGecko强调去中心化与多维评估,助力项目潜力分析;3. TradingView集成实时行情与技术工具,满足专业图表分析需求;4. Dune Analytics通过SQL查询链上数据,实现深度业务洞察;5. Def…

    2025年12月11日
    000
  • 什么是链上随机数?它为何难以生成且至关重要?

    链上随机数是通过去中心化方式生成不可预测数值的机制,用于确保智能合约执行的公平性。由于区块链的确定性特性,直接生成安全随机数困难,主要面临矿工操纵、缺乏熵源及算法可重现等问题。为解决这些挑战,常用方法包括:采用链下预言机如Chainlink VRF提供带加密证明的随机数,利用未来区块哈希作为延迟随机…

    2025年12月11日
    000
  • 为什么你总是拿不住币?这套心态管理法让你稳如泰山!

    建立持仓原则、控制查看频率、重构认知、构建反馈机制是稳定心态的关键。明确买入逻辑并记录依据,设定不可违背的规则如“未达目标不卖出”,并将纪律写入备忘录;减少盯盘,移除行情软件主屏、每日固定时间查看一次、关闭价格推送;下跌时问是否影响底层价值,改黑白K线图,卖出前写三个持有理由;设持仓里程碑奖励自己,…

    2025年12月11日
    000
  • Janction (JCT)币应用场景_JCT长期价值预测

    Janction(JCT)是融合区块链与AI的去中心化计算平台,1. 构建分布式GPU算力市场,用户注册并连接GPU设备后加入算力池,需求方通过智能合约提交任务,系统分配至vGPU节点处理,完成后按贡献分发JCT代币;2. 支持多方协同训练AI模型,发起方加密发布任务,节点本地计算并提交结果及零知识…

    2025年12月11日
    000
  • 永续合约资金费率是什么?为什么做多有时需要给做空付钱

    资金费率是永续合约中连接合约与现货价格的机制,当合约价高于现货价时,多头向空头支付费用以平衡价格偏离,抑制过度看涨情绪。正值资金费率反映市场溢价,做多成本因此增加,交易者可通过平仓、套利、选择低费率品种或避开结算时点等方式管理费用风险。 binance币安交易所 注册入口: APP下载: 欧易OKX…

    2025年12月11日
    000
  • Janction (JCT)币项目深度研究_JCT币价目标预测

    JCT代币的核心价值源于其去中心化AI算力网络的构建,白皮书明确了技术路径与长期目标,团队背景待核实,项目已有测试网运行,当前流通量115亿,占总量23%,释放节奏相对平稳;市场交易集中于CoinEx等平台,JCT/USDT交易对成交活跃,近期价量齐升显示资金关注度提高,但需警惕高换手率带来的波动风…

    2025年12月11日
    000
  • Janction (JCT)币发展路线图_JCT价格预测模型

    Janction主网上线并提升网络稳定性,集成AI计算市场功能,部署跨链互操作性协议,上线治理系统与社区决策机制,开放企业级API接口。 Janction (JCT) 是一个结合 DePIN 与 AI 的区块链项目,旨在构建去中心化的人工智能计算基础设施。 一、主网上线与网络稳定性提升 该阶段的核心…

    2025年12月11日
    000

发表回复

登录后才能评论
关注微信