最新资讯

  • 【Linux实践系列】:用c语言实现一个shell外壳程序

【Linux实践系列】:用c语言实现一个shell外壳程序

2025-04-26 23:37:12 0 阅读

🔥本文专栏:Linux Linux实践项目
🌸博主主页:努力努力再努力wz



那么今天我们就要进入Linux的实践环节,那么我们之前学习了进程控制相关的几个知识点,比如进程的终止以及进程的等待和进程的替换,那么我们接下来就要结合前面所讲的进程控制相关的接口比如fork以及waitpid和execl等来自己实现一个命令行解释器,那么废话不多说,让我们进入正文

★★★ 本文前置知识
进程的替换
进程的终止与等待
进程的概念


shell实现的框架

那么在用c语言真正上手实操我们的shell外壳程序时,那么我们脑海里得有一个大体的实现框架,也就是所谓的一个整体思路,在有了整体思路后我们再去谈具体每个模块的细节,那么我们首先就先从shell本身的工作原理作为切入口入手

那么我们的shell也就是我们的命令行解释器,那么它的工作就是获取用户输入的指令,然后来执行用户输入的指令,那么我们知道用户输入的指令的本质上就是一个字符串,所以shell首先就得读取到用户输入的字符串,然后保存在一个字符数组中,读取到用户输入的字符串之后,那么紧接着下一步便是解析用户输入的字符串,那么我们用户输入的指令无非可以分成两大部分,分别是指令部分以及参数部分,那么这里我们就需要定义一个字符指针数组,那么数组的每一个元素就是一个指针,那么指针指向的就是一个字符串,那么我们用户输入的字符串的指令部分就保存在字符指针数组的第一个元素也就是下标为0的位置,那么参数部分则依次保存在之后的位置,比如我们用户输入的指令是ls -l -a,那么此时我们就要解析为三部分,分别是是指令部分的“ls”字符串以及两个参数部分的字符串"-l”和“-a”,将这三个字符串则是依次保存到我们的字符指针数组下标为0和1和2的位置当中

而具体的解析这三部分字符串则需要用到我们c语言的strtok函数,那么具体细节我们下文再说,那么这里我们讨论的是大的框架与思路,所以我们可以专门定义一个函数来完成这个字符串解析的模块,它的工作就是解析用户输入的字符串将其指令部分以及参数部分的各个字符串分别保存到字符指针数组不同位置中,并且返回命令行的个数,比如用户输入的是ls -l,那么将其保存在字符指针数组char* argv[]并返回的个数就是2,,而如果是pwd,将其保存在字符指针数组char* argv[]并返回的个数就是1

那么接下来解析完用户输入的字符串之后,那么我们就可以来执行用户输入的指令了,那么这里我们知道我们用户输入的各种指令本质上就是在特定路径下保存的一个可执行文件,那么指令的执行本质上就是创建一个进程,那么我们shell执行这些指令就得利用fork函数来创建一个子进程,然后我们利用fork函数的返回值,将父子进程分成不同的执行流,那么在子进程的执行流代码片段中,我们就可以利用进程的替换,那么将我们的子进程的内容替换为我们要执行指令所对应的进程的上下文,那么我们父进程的执行流代码片段则是等待我们子进程的退出结果,那么我们就需要用waitpid函数来获取子进程的退出码

最后获取完子进程的退出码,如果子进程没有正常终止,那么就得将情况返回给用户,也就是将错误信息打印到终端,如果子进程正常终止然后下一步就是重复我们之前上文的环节,那么重复也就意味着我们实现的时候最后这些逻辑的代码都要封装到一个死循环当中。

那么这就是我们实现shell外壳程序的一个大框将,那么我们可以简单将其分为几个模块,分别是获取用户输入->解析用户输入->创建子进程->子进程的替换->父进程等待获取子进程的退出情况->重复上述步骤

那么看到这些模块,想必你一定还有一些疑问,那么接下来我就会在下文补充每个模块的代码实现以及注意的一些细节,和其他的模块的补充,那么有了大框架之后,那么接下来就让我们具体实现每一个模块了

shell各个模块的实现

1.获取用户输入

那么我们的shell首先得获取用户输入的字符串,那么我们知道在c语言中,我们获取用户输入的字符串常见就是使用我们的scanf函数来获取用户的输入,但是scanf函数有一个缺陷就是一旦读取到空格的时候,那么scanf便停止读取输入,而我们用户在输入字符串的时候,会手动用空格隔开指令部分与参数部分,所以我们就不能采取scanf函数来获取输入,所以这里我们需要用fgets函数,那么fgets函数则是将从标准输入流中读取用户的输入,遇到换行符停止,那么我们可以指定其在输入流中读取的字符串的长度也就是fgets的第二个参数,那么将其保存到一个临时字符数组中,如果读取失败,那么fgets则会返回NULL,读取成功fgets则会返回保存数组的地址

  • fgets

    头文件:

  • 功能:获取用户输入的字符串,末尾自动添加

    :函数原型

char *fgets(char *str, int n, FILE *stream);

而我们知道用户在输入之前,我们终端都会显示一个命令行提示符,会显示我们当前登录的用户名以及所处的工作目录和运行的主机名称,所以我们在获取用户输入之前,我们得先打印一个字符串也就是命令行提示符,而切记,我们的shell命令行解释器本质也是一个进程,所以这命令行提示符的每一个信息就保存在我们当前进程的环境变量中,我们需要通过我们的系统调用接口getenv来获取其中特定字段的环境变量,这里就需要获取到我们的USER以及HOSTNMAE以及PWD这三个字段,那么我们只需要向getenv函数传递这三个字符串的指针,那么他会依次匹配各个字段的名称所对应的字符串并返回对应的值,也就是字符串的起始地址

代码实现:

  printf("[%s@%s %s]$",getenv("USER"),getenv("HOSTNAME"),getenv("PWD"));
	  	  if(fgets(temp,sizeof(temp),stdin)==NULL)
	  	  {
	  	  	  perror("fgets");
	  	  	  continue;
			}

2.解析命令行

那么现在我们获取了我们的用户输入的字符串之后,那么我们是将用户输入的字符串保存在一个临时字符数组里,那么接下来我们就要将这个字符串给分割,将其指令部分以及参数部分的各个字符串给分割保存到我们的字符指针数组当中,那么我们这里就专门可以定义一个函数来完成字符串解析模块,并返回命令行参数的个数,那么我们知道我们用户输入的字符串会手动以空格分割,那么这里我们就需要调用我们的字符串函数也就是strtok函数来分割我们的字符串按照空格作为分隔符。但是在分割之前,我们又得注意一个细节,也就是我们用户输入完一个字符串,那么它会敲一个回车键来表示输入的结束,而回车则是对应的一个换行符 ,他会被我们的fegts给读取到,那么意味着在我们的字符串的末尾可能会有一个回车换行符

而回车换行符并不是我们一个有效的字符信息,所以我们在解析之前得去掉这个换行符,所以我们就利用我们的strlen函数首先获取到我们这个字符串包括空格以及换行符的总长度,那么如果我判断用户输入的字符串的len-1位置处的字符处是换行符,那么我们就将len-1位置用来覆盖,而是标记字符串结尾的标志,那么这样我们就可以消去末尾的回车换行符,这里是其中一个关键的实现细节

那么第二个细节就是我们的strtok函数的使用,那么我们strtok函数第一次调用的时候要传递我们要分割的字符串的首元素的地址,那么strtok内部会访问到一个静态的全局变量,这个静态变量是用来保存下一次分割的位置,那么我们每次调用strtok函数的时候,会从分割的起始位置处往后扫描直到遇到分隔符,然后将分隔符的位置修改为,然后返回该分割起始位置的指针,而我们知道是标记字符串的结尾,所以返回分割起始位置的指针就达到了一个分割子串的一个效果
而下一次调用strtok函数的时候,那么我们就不用传要分割的字符串的首元素的地址,因为上文说过strtok内部能访问到一个记录下一次分割位置的全局变量,那么之后的调用我们只需要传递一个NULL即可,它内部会继续从这个全局变量记录的位置开始扫描到下一个分隔符,将其修改为,最后如果我们开始的分割的位置是,也就是字符串末尾,没有更多的子串来分割时候,那么strtok就返回一个NULL


  • strtok
    头文件:
    -功能:分割字符串

    :函数原型

 char *strtok(char *str, const char *delim);

在这个函数中我们就定义一个int类型的argc变量来跟踪命令行的个数,初始化为0,而我们将分割的字符串保存在对应的字符数组的下标就是argc的值,保存之后接着递增argc,最后返回的该argc就是我们的命令行的个数

代码实现:

int getString(char temp[],char* argv[])
{
       int len=strlen(temp);
       if(len>0&&temp[len-1]=='
')
       {
           temp[len-1]='';
           len--;
       }
       int argc=0;
       char* toke=strtok(temp," ");
       while(toke!=NULL&&argc<length-1)
       {
              argv[argc++]=toke;
             toke=strtok(NULL," ");
       }
       argv[argc]=NULL;
       return argc;
}

3.指令判断

那么这里在我们上文介绍实现我们的shell外壳程序的框架的时候模块的时候,其实我们故意漏了一个模块,那就是指令的判断,那么想必你一定会有所疑问,那么就是我们获取解析完用户输入的指令之后,我们为什么还要进行指令的判断呢?直接通通交给子进程去执行不就完了吗,我们父进程也就是shell外壳程序的本职工作不就是获取用户的输入吗

那么这里我们就要注意的就是,我们用户其中输入的指令,比如cd指令,也就是更改我们进程所处的工作目录,那么它针对的对象其实是我们的父进程也就是我们的shell外壳程序,那么如果我们把这个指令交给了子进程去完成,将子进程替换为cd指令所对应的上下文,那么子进程的执行是不会影响父进程的,那么子进程执行结束退出之后,我们shell进程所处的工作目录没有进行更改,那么所以我们对于有些指令,也就是针对当前父进程shell的运行环境的指令,比如cd,比如PWD指令,那么它就不能交给子进程来执行,而是得交给父进程来自己完成,那么这些指令也就是我们的内置指令

那么内置指令那么就不再是一个编写好的可执行文件,那么它是通常是一个实现好的库函数或者直接嵌套在我们的shell进程所对应的代码中,所以我们自己用c语言实现的时候,那么我们就首先准备定义一个全局属性的字符指针数组,然后该数组里面记录了我们所有的内置命令所对应的字符串,那么当解析完用户的指令之后,解析完保存的字符指针数组的第一个位置就是对应用户输入的指令部分的字符串,所以下一步我们依次匹配保存的所有内置命令对应的字符串,如果匹配成功,那么意味着是内置指令,就直接交给我们父进程执行,匹配失败则说明该指令不是内置命令,就交给子进程来执行,那么我们匹配的过程以及内置命令的执行的过程都可以定义两个函数来分别实现这两个模块,那么其中字符串的匹配就需要用到strcmp函数来实现

而所谓的内置命令,他的底层实现的时候本质其实就是依赖用c编写的库函数或者系统调用,比如cd内置命令,那么它就是用chdir库函数来实现的,那么这个库函数的作用就是能够访问到当前进程的环境变量中的工作目录字段,然后修改当前所处的工作目录,而pwd内置命令的本质其实也就是依赖getcwd库函数,那么该库函数会访问到该进程中环境变量记录当前所处也就是工作目录的字段PWD,将其值记录保存到一个数组当中,并且返回指向该数组的指针

  • chdir
    头文件:

    :函数原型

  int chdir(const char *path);
  
  • getcwd
    头文件:

    :函数原型

    char *getcwd(char *buf, size_t size);
  
  

那么这里我在实现的时候,就只判断了cd以及pwd这两种内置命令,那么我们可以下来直接去添加更多的内置命令,然后查询对应实现所依赖的库函数或者系统调用接口

代码实现:

bool check(char* argv[])//指令的判断
{
    for(int i=0;order[i]!=NULL;i++)
    {
         if(strcmp(argv[0],order[i])==0)//如果该指令是内置命令就返回true
         {
             return true;
         }
    }
      return false;
}
void ordercomplete(int argc,char* argv[])//内置命令的执行
{
       if(strcmp(argv[0],"cd")==0)
       {
                if(argc==2)//cd指令最多只能两个参数,其中第二个参数就是跳转的工作目录
                {
               if (chdir(argv[1]) != 0) {
                perror("chdir");
            }
                }else
                {
                    printf("error: expected argument for 'cd'
");
                }
       }
       if(strcmp(argv[0],"pwd")==0)
       {
              char cwd[length];  // 定义一个字符数组错误的来保存我们的当前所处的工作目录

    if (getcwd(cwd, sizeof(cwd)) != NULL) {
        printf("Current working directory: %s
", cwd);
    } else {
        perror("getcwd failed");  // 输出错误信息
    }
       }
}

5.子进程执行指令

那么剩下几个模块的细节和实现就很简单了,接下来这个模块就调用fork函数来创建一个子进程,然后利用fork函数的返回值,让父子进程有着不同的执行流,然后我们在子进程对应的执行流代码片段中,调用进程的替换的系统接口,而这里我们调用的exec族函数,一定是不能带有l的比如execl以及execlp等,因为我们不知道用户输入的命令行个数,所以不能用可变参数列表的进程替换接口,这里要注意
而我们用户输入的字符串都解析在了一个字符指针数组中,所以我们传的参数肯定就是一个数组,所以这里我们选择进程替换的函数就是execvp,那么它可以默认在环境变量的PATH中去匹配我们用户输入的指令所对应的可执行文件
那么我们用execvp函数来将子进程替换为指令所对应的进程的上下文,但是我们知道我们进程替换会出现调用失败的情况,那么调用失败的结果则是会执行进程替换接口之后的代码,那么我们就在execvp函数后面打印一个错误信息并且返回一个特殊的退出码

6.父进程的等待

那么我们父进程对应的执行流代码片段则是等待我们子进程的退出情况,所以我们需要调用waitpid函数来获取子进程的退出码,那么waitpid我们的等待方式则是设置为阻塞式等待,那么它的返回值就分别对应两种情况,要么等待成功并且获取到子进程的退出码,对应的返回值就是子进程的pid,而等待失败则是返回-1,我们对于等待失败则是要打印错误信息以及子进程的退出码

完整实现

那么将我们上面的6个模块所对应代码融合就是我们的shell的外壳程序,那么其实我们在实现shell外壳程序的时候,其实shell的整体实现难度不大,主要考察你对shell的工作原理的理解程度和几个系统调用接口的熟悉程度,shell实现的真正的难点其实在它各个模块实现的细节上,很容易出错,其中就考察我们对于一些c语言的库函数的掌握情况,那么接下来我就给出完成的shell的c语言代码的实现

#include
#include
#include
#include
#include
#include
#include
#define length 1000
#define EXIT_FAIL 40
const char* order[]={"cd","pwd",NULL};

int getString(char temp[],char* argv[])
{
       int len=strlen(temp);
       if(len>0&&temp[len-1]=='
')
       {
           temp[len-1]='';
           len--;
       }
       int argc=0;
       char* toke=strtok(temp," ");
       while(toke!=NULL&&argc<length-1)
       {
              argv[argc++]=toke;
             toke=strtok(NULL," ");
       }
       argv[argc]=NULL;
       return argc;
}
bool check(char* argv[])
{
    for(int i=0;order[i]!=NULL;i++)
    {
         if(strcmp(argv[0],order[i])==0)
         {
             return true;
         }
    }
      return false;
}
void ordercomplete(int argc,char* argv[])
{
       if(strcmp(argv[0],"cd")==0)
       {
                if(argc==2)
                {
               if (chdir(argv[1]) != 0) {
                perror("chdir");
            }
                }else
                {
                    printf("error: expected argument for 'cd'
");
                }
       }
       if(strcmp(argv[0],"pwd")==0)
       {
              char cwd[length];  // 定义一个足够大的缓冲区来存储路径

    if (getcwd(cwd, sizeof(cwd)) != NULL) {
        printf("Current working directory: %s
", cwd);
    } else {
        perror("getcwd failed");  // 输出错误信息
    }
       }
}
int main()
{
      int argc;
      char* argv[length];
      char temp[length];
      
      while(1)
      {
            printf("[%s@%s %s]$",getenv("USER"),getenv("HOSTNAME"),getenv("PWD"));
            if(fgets(temp,sizeof(temp),stdin)==NULL)
            {
                  perror("fgets");
                  continue;
            }
            argc=getString(temp,argv);
            if(argc==0)
            {
                continue;
            }
            if(check(argv))
            {
                ordercomplete(argc,argv);
                continue;
            }
            int id=fork();
            if(id==0)
            {
                execvp(argv[0],argv);
                perror("execvp");
                exit(EXIT_FAIL);
            }else
            {
                int status;
                int m=waitpid(id,&status,0);
                if(m<0)
                {
                    perror("waitpid");
                }else
                {
                     if(WIFEXITED(status))
                     {
                           if(WEXITSTATUS(status)==40)
                           {
                                printf("error
");
                           }
                     }
                }
                
            }
      }
      return 0;
}

在Linux上的运行截图:

结语

那么这就是用c语言实现shell外壳程序的所有内容啦,那么它也是我第一个学习Linux所完成的一个小项目,那么它这个小项目的教学价值以及学习意义其实非常高,因为它不仅可以帮组你了解shell外壳程序的工作原理,更重要的是帮组你更能熟练掌握运用那几个关于进程控制十分重要的系统调用接口其中比如fork以及waitpid等,那么我的下一篇Linux文章就正式进入文件系统啦,我会持续更新,希望大家多多关注,那么如果本篇文章对你有所帮组的话,那么还请多多三连加关注哦,你的支持就是我最大的动力!

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

搜索文章

Tags

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