当前位置: 首页 > news >正文

模拟shell小程序

接下来利用我们当前的知识,撰写一个简单的shell外壳程序。

1.shell原理

shell的原理是实际上就是运行了一个父进程,然后创建出子进程,最后使用进程替换调用,替换成其他程序。

2.shell实现

2.1.死循环

首先一个shell一旦运行起来就不会关闭(除非关闭终端窗口),因此一定是一个死循环。

int main()
{while(1){//...}return 0}

2.2.提示字符串

运行一个shell有出现一个类似[user@myshell]$的输入提示符,由于不能换行,因此我们还需要使用fflush()进行刷新。

#include <stdio.h>
int main()
{while(1){printf("[user@myshell]$ ");fflush(stdout);}return 0;
}

2.3.获取用户输入

#include <stdio.h>
#include <string.h>#define NUM 1024
char cmd_line[NUM];//保存完整命令字符串int main()
{while(1){//1.打印提示信息printf("[user@myshell]$ ");fflush(stdout);//2.获取 user 的输入memset(cmd_line, '\0', sizeof(cmd_line));//设定初始值if(fgets(cmd_line, sizeof(cmd_line), stdin) == NULL)//获取字符串{//出错就直接跳出循环continue;}cmd_line[strlen(cmd_line) - 1] = '\0';//即使用户输入字符串超出范围,也能保证获取到一个完整的 C 风格字符串}return 0;
}

2.4.解析用户输入

#include <stdio.h>
#include <string.h>#define NUM 1024
char cmd_line[NUM];//保存完整命令字符串#define SIZE 32
char* g_argv[SIZE];//保存打散后的命令字符串,是一个数组#define SEP " "//分隔符int main()
{while(1){//1.打印提示信息printf("[user@myshell]$ ");fflush(stdout);//2.获取 user 的输入memset(cmd_line, '\0', sizeof(cmd_line));//设定初始值if(fgets(cmd_line, sizeof(cmd_line), stdin) == NULL)//获取字符串{//出错就直接跳出循环continue;}cmd_line[strlen(cmd_line) - 1] = '\0';//即使用户输入字符串超出范围,也能保证获取到一个完整的 C 风格字符串//3.命令行字符解析//虽然可以自己写一个算法解析,但是我们可以使用一些现有的接口g_argv[0] = strtok(cmd_line, SEP);//第一次调用需要传入原始字符串和分隔符int index = 1;if(strcmp(g_argv[0], "ls") == 0)//如果输入的命令是 ls 就进行一些特殊处理{g_argv[index++] = "--color=auto";//增加颜色参数,让输出带有颜色}while(g_argv[index++] = strtok(NULL, SEP));//第二次还想要解析原始字符串,就需要传入 NULL}return 0;
}

2.5.创建并且替换子进程

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/types.h> 
#include <string.h>
#include <stdlib.h>#define NUM 1024
char cmd_line[NUM];//保存完整命令字符串#define SIZE 32
char* g_argv[SIZE];//保存打散后的命令字符串,是一个数组#define SEP " "//分隔符int main()
{while(1){//1.打印提示信息printf("[user@myshell]$ ");fflush(stdout);//2.获取 user 的输入memset(cmd_line, '\0', sizeof(cmd_line));//设定初始值if(fgets(cmd_line, sizeof(cmd_line), stdin) == NULL)//获取字符串{//出错就直接跳出循环continue;}cmd_line[strlen(cmd_line) - 1] = '\0';//即使用户输入字符串超出范围,也能保证获取到一个完整的 C 风格字符串//3.命令行字符解析//虽然可以自己写一个算法解析,但是我们可以使用一些现有的接口g_argv[0] = strtok(cmd_line, SEP);//第一次调用需要传入原始字符串和分隔符int index = 1;if(strcmp(g_argv[0], "ls") == 0)//如果输入的命令是 ls 就进行一些特殊处理{g_argv[index++] = "--color=auto";//增加颜色参数,让输出带有颜色}while(g_argv[index++] = strtok(NULL, SEP));//第二次还想要解析原始字符串,就需要传入 NULL//4.fork() 创建子进程,execvp() 替换子进程pid_t id = fork();if(id == 0)//child{printf("子进程 run:\n");execvp(g_argv[0], g_argv);//自动在环境变量中搜索 g_argv[0] 的文件,执行 g_argv 数组存储的指令exit(1);//没替换成功就会异常退出}//fatherint status;pid_t ret = waitpid(id, &status, 0);//阻塞等待if(ret > 0) printf("exit code: %d\n", WEXITSTATUS(status));}return 0;
}

2.6.debug

有一个小小的bug我们没有解决,就是使用cd命令的时候,没有办法切入到对应的目录,这是为什么呢?

3.shell拓展

3.1.用户名、主机名、目录获取

上面我们只是生硬显示了user充当用户名字,实际上我们可以通过环境变量获取执行shell的用户的名字,还可以获取在不同环境下的系统主机名字,使用这两个环境变量的内容填充到提示字符串中,目录也可以这样操作。

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/types.h> 
#include <string.h>
#include <stdlib.h>#define NUM 1024
char cmd_line[NUM];//保存完整命令字符串#define SIZE 32
char* g_argv[SIZE];//保存打散后的命令字符串,是一个数组#define SEP " "//分隔符const char* GetUserName()//获取用户名字
{const char* userName = getenv("USER");if(userName) return userName;elsereturn "none";
}const char* GetHostName()//获取主机名字
{const char* hostName = getenv("HOSTNAME");if(hostName)return hostName;elsereturn "none";
}const char* GetPwd()//获取路径地址
{const char* pwd = getenv("PWD");if(pwd)return pwd;elsereturn "none";
}int main()
{while(1){//1.打印提示信息printf("[%s@%s %s]$ ", GetUserName(), GetHostName(), GetPwd());fflush(stdout);//2.获取 user 的输入memset(cmd_line, '\0', sizeof(cmd_line));//设定初始值if(fgets(cmd_line, sizeof(cmd_line), stdin) == NULL)//获取字符串,不能使用 scanf(){//出错就直接跳出循环continue;}cmd_line[strlen(cmd_line) - 1] = '\0';//将用户输入的字符串去掉 \n//3.命令行字符解析//虽然可以自己写一个算法解析,但是我们可以使用一些现有的接口g_argv[0] = strtok(cmd_line, SEP);//第一次调用需要传入原始字符串和分隔符int index = 1;if(strcmp(g_argv[0], "ls") == 0)//如果输入的命令是 ls 就进行一些特殊处理{g_argv[index++] = "--color=auto";//增加颜色参数,让输出带有颜色}while(g_argv[index++] = strtok(NULL, SEP));//第二次还想要解析原始字符串,就需要传入 NULL//4.fork() 创建子进程,execvp() 替换子进程pid_t id = fork();if(id == 0)//child{printf("子进程 run:\n");execvp(g_argv[0], g_argv);//自动在环境变量中搜索 g_argv[0] 的文件,执行 g_argv 数组存储的指令exit(1);//没替换成功就会异常退出}//fatherint status;pid_t ret = waitpid(id, &status, 0);//阻塞等待if(ret > 0) printf("exit code: %d\n", WEXITSTATUS(status));}return 0;
}

3.2.封装步骤

既然上面的用户名、主机名、路径都进行了函数封装,那么我也可以选择将部分的代码进行分钟,变得更有逻辑。

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/types.h> 
#include <string.h>
#include <stdlib.h>
#include <stdbool.h>#define NUM 1024
char cmd_line[NUM];//保存完整命令字符串#define SIZE 32
char* g_argv[SIZE];//保存打散后的命令字符串,是一个数组#define SEP " "//分隔符const char* GetUserName(void)//获取用户名字
{const char* userName = getenv("USER");if(userName) return userName;elsereturn "none";
}const char* GetHostName(void)//获取主机名字
{const char* hostName = getenv("HOSTNAME");if(hostName)return hostName;elsereturn "none";
}const char* GetPwd(void)//获取路径地址
{const char* pwd = getenv("PWD");if(pwd)return pwd;elsereturn "none";
}void Print(void)
{printf("[%s@%s %s]$ ", GetUserName(), GetHostName(), GetPwd());fflush(stdout);
}bool GetUserCommand()
{memset(cmd_line, '\0', sizeof(cmd_line));//设定初始值if(fgets(cmd_line, sizeof(cmd_line), stdin) == NULL)//获取字符串,不能使用 scanf(){//出错就直接跳出循环return false;}cmd_line[strlen(cmd_line) - 1] = '\0';//将用户输入的字符串去掉 \nreturn true;
}void AnalyzeStr(void)
{//虽然可以自己写一个算法解析,但是我们可以使用一些现有的接口g_argv[0] = strtok(cmd_line, SEP);//第一次调用需要传入原始字符串和分隔符int index = 1;//处理特殊的命令情况if(strcmp(g_argv[0], "ls") == 0)//如果输入的命令是 ls 就进行一些特殊处理{g_argv[index++] = "--color=auto";//增加颜色参数,让输出带有颜色}while(g_argv[index++] = strtok(NULL, SEP));//第二次还想要解析原始字符串,就需要传入 NULL
}void RunCommand(void)
{pid_t id = fork();if(id == 0)//child{printf("子进程 run:\n");execvp(g_argv[0], g_argv);//自动在环境变量中搜索 g_argv[0] 的文件,执行 g_argv 数组存储的指令exit(1);//没替换成功就会异常退出}else{//fatherint status;pid_t ret = waitpid(id, &status, 0);//阻塞等待if(ret > 0) printf("exit code: %d\n", WEXITSTATUS(status));}
}int main()
{while(1){//1.打印提示信息Print();//2.获取 user 的输入if(!GetUserCommand())continue;//3.命令行字符解析AnalyzeStr();//4.fork() 创建子进程,execvp() 替换子进程RunCommand();}return 0;
}

3.3.debug

如果使用了上面的shell程序,很快就会发现两个bug

3.3.1.无法直接输入换行

在系统默认运行的shell中,允许用户不断回车,而我们的程序如果直接回车就会退出,这个问题的解决思路很简单,只需要在获取字符的时候,查看用户输入的字符长度是否为0即可,若是就让GetUserCommand()返回false,执行continue语句即可。

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/types.h> 
#include <string.h>
#include <stdlib.h>
#include <stdbool.h>#define NUM 1024
char cmd_line[NUM];//保存完整命令字符串#define SIZE 32
char* g_argv[SIZE];//保存打散后的命令字符串,是一个数组#define SEP " "//分隔符const char* GetUserName(void)//获取用户名字
{const char* userName = getenv("USER");if(userName) return userName;elsereturn "none";
}const char* GetHostName(void)//获取主机名字
{const char* hostName = getenv("HOSTNAME");if(hostName)return hostName;elsereturn "none";
}const char* GetPwd(void)//获取路径地址
{const char* pwd = getenv("PWD");if(pwd)return pwd;elsereturn "none";
}void Print(void)
{printf("[%s@%s %s]$ ", GetUserName(), GetHostName(), GetPwd());fflush(stdout);
}bool GetUserCommand()
{memset(cmd_line, '\0', sizeof(cmd_line));//设定初始值if(fgets(cmd_line, sizeof(cmd_line), stdin) == NULL)//获取字符串,不能使用 scanf(){//出错就直接跳出循环return false;}cmd_line[strlen(cmd_line) - 1] = '\0';//将用户输入的字符串去掉 \nif(strlen(cmd_line) == 0)return false;return true;
}void AnalyzeStr(void)
{//虽然可以自己写一个算法解析,但是我们可以使用一些现有的接口g_argv[0] = strtok(cmd_line, SEP);//第一次调用需要传入原始字符串和分隔符int index = 1;//处理特殊的命令情况if(strcmp(g_argv[0], "ls") == 0)//如果输入的命令是 ls 就进行一些特殊处理{g_argv[index++] = "--color=auto";//增加颜色参数,让输出带有颜色}while(g_argv[index++] = strtok(NULL, SEP));//第二次还想要解析原始字符串,就需要传入 NULL
}
void RunCommand(void)
{pid_t id = fork();if(id == 0)//child{printf("子进程 run:\n");execvp(g_argv[0], g_argv);//自动在环境变量中搜索 g_argv[0] 的文件,执行 g_argv 数组存储的指令exit(1);//没替换成功就会异常退出}else{//fatherint status;pid_t ret = waitpid(id, &status, 0);//阻塞等待if(ret > 0) printf("exit code: %d\n", WEXITSTATUS(status));}
}int main()
{while(1){//1.打印提示信息Print();//2.获取 user 的输入if(!GetUserCommand())continue;//3.命令行字符解析AnalyzeStr();//4.fork() 创建子进程,execvp() 替换子进程RunCommand();}return 0;
}

3.3.2.无法切换对应目录

还可以发现一个比较严重的问题,我们无法使用cd命令切换目录,这是为什么呢?原因也很简单,cd程序替换的是子进程,该目录只让子进程进行了切换命令,而我们希望的是父进程自己执行cd,修改父进程的工作路径。因此,我们需要调用一些库函数,更改父进程的工作环境。

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/types.h> 
#include <string.h>
#include <stdlib.h>
#include <stdbool.h>#define NUM 1024
char cmd_line[NUM];//保存完整命令字符串#define SIZE 32
char* g_argv[SIZE];//保存打散后的命令字符串,是一个数组#define SEP " "//分隔符const char* GetUserName(void)//获取用户名字
{const char* userName = getenv("USER");if(userName) return userName;elsereturn "none";
}const char* GetHostName(void)//获取主机名字
{const char* hostName = getenv("HOSTNAME");if(hostName)return hostName;elsereturn "none";
}const char* GetPwd(void)//获取路径地址
{const char* pwd = getenv("PWD");if(pwd)return pwd;elsereturn "none";
}void Print(void)//打印提示字符
{printf("[%s@%s %s]$ ", GetUserName(), GetHostName(), GetPwd());fflush(stdout);
}bool GetUserCommand(void)//获取用户命令字符串
{memset(cmd_line, '\0', sizeof(cmd_line));//设定初始值if(fgets(cmd_line, sizeof(cmd_line), stdin) == NULL)//获取字符串,不能使用 scanf(){//出错就直接跳出循环return false;}cmd_line[strlen(cmd_line) - 1] = '\0';//将用户输入的字符串去掉 \nif(strlen(cmd_line) == 0)return false;return true;
}void AnalyzeStr(void)//拆分解析用户字符串
{//虽然可以自己写一个算法解析,但是我们可以使用一些现有的接口g_argv[0] = strtok(cmd_line, SEP);//第一次调用需要传入原始字符串和分隔符int index = 1;//处理特殊的命令情况if(strcmp(g_argv[0], "ls") == 0)//如果输入的命令是 ls 就进行一些特殊处理{g_argv[index++] = "--color=auto";//增加颜色参数,让输出带有颜色}while(g_argv[index++] = strtok(NULL, SEP));//第二次还想要解析原始字符串,就需要传入 NULL
}bool DoBuildin(void)//判断是否内建命令并且执行
{//内建命令if(strcmp(g_argv[0], "cd") == 0)       {   printf("父进程 run:\n");//更改路径的命令char* path = NULL;if(g_argv[1] == NULL){path = ".";}else{path = g_argv[1];}chdir(path);return true;}return false;
}void RunCommand(void)//运行用户命令
{pid_t id = fork();if(id == 0)//child{printf("子进程 run:\n");execvp(g_argv[0], g_argv);//自动在环境变量中搜索 g_argv[0] 的文件,执行 g_argv 数组存储的指令exit(1);//没替换成功就会异常退出}else{//fatherint status;pid_t ret = waitpid(id, &status, 0);//阻塞等待if(ret > 0) printf("exit code: %d\n", WEXITSTATUS(status));}
}int main()
{while(1){//1.打印提示信息Print();//2.获取 user 的输入if(!GetUserCommand())continue;//3.命令行字符解析AnalyzeStr();//4.处理内建命令if(DoBuildin())continue;//5.fork() 创建子进程,execvp() 替换子进程RunCommand();}return 0;
}

这种只能由父进程来执行的命令也叫做“内建命令”,这类环境变量有很多。

3.3.3.无法更改环境变量

还有一个比较尴尬的问题,我们发现使用cd指令后,提示字符串的地址无法及时更新,也就是没有更新对应的环境变量。如果更新呢?可以使用系统提供的接口getcwd()来获取调用该接口进程所在的路径。

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/types.h> 
#include <string.h>
#include <stdlib.h>
#include <stdbool.h>#define NUM 1024
char cmd_line[NUM];//保存完整命令字符串#define SIZE 32
char* g_argv[SIZE];//保存打散后的命令字符串,是一个数组#define SEP " "//分隔符char cwd[1024];//环境变量const char* GetUserName(void)//获取用户名字
{const char* userName = getenv("USER");if(userName) return userName;elsereturn "none";
}const char* GetHostName(void)//获取主机名字
{const char* hostName = getenv("HOSTNAME");if(hostName)return hostName;elsereturn "none";
}const char* GetPwd(void)//获取路径地址
{const char* pwd = getenv("PWD");if(pwd)return pwd;elsereturn "none";
}void Print(void)
{printf("[%s@%s %s]$ ", GetUserName(), GetHostName(), GetPwd());fflush(stdout);
}bool GetUserCommand()
{memset(cmd_line, '\0', sizeof(cmd_line));//设定初始值if(fgets(cmd_line, sizeof(cmd_line), stdin) == NULL)//获取字符串,不能使用 scanf(){//出错就直接跳出循环return false;}cmd_line[strlen(cmd_line) - 1] = '\0';//将用户输入的字符串去掉 \nif(strlen(cmd_line) == 0)return false;return true;
}void AnalyzeStr(void)
{//虽然可以自己写一个算法解析,但是我们可以使用一些现有的接口g_argv[0] = strtok(cmd_line, SEP);//第一次调用需要传入原始字符串和分隔符int index = 1;//处理特殊的命令情况if(strcmp(g_argv[0], "ls") == 0)//如果输入的命令是 ls 就进行一些特殊处理{g_argv[index++] = "--color=auto";//增加颜色参数,让输出带有颜色}while(g_argv[index++] = strtok(NULL, SEP));//第二次还想要解析原始字符串,就需要传入 NULL
}bool DoBuildin()
{//内建命令if(strcmp(g_argv[0], "cd") == 0)       {   printf("父进程 run:\n");//更改路径的命令char* path = NULL;if(g_argv[1] == NULL){path = ".";}else{path = g_argv[1];}chdir(path);char tmp[1024];getcwd(tmp, sizeof(tmp));//获取当前进程所在的地址(工作目录)sprintf(cwd, "PWD=%s", tmp);//组合环境变量putenv(cwd);//设置环境变量return true;}return false;
}void RunCommand(void)
{pid_t id = fork();if(id == 0)//child{printf("子进程 run:\n");execvp(g_argv[0], g_argv);//自动在环境变量中搜索 g_argv[0] 的文件,执行 g_argv 数组存储的指令exit(1);//没替换成功就会异常退出}else{//fatherint status;pid_t ret = waitpid(id, &status, 0);//阻塞等待if(ret > 0) printf("exit code: %d\n", WEXITSTATUS(status));}
}int main()
{while(1){//1.打印提示信息Print();//2.获取 user 的输入if(!GetUserCommand())continue;//3.命令行字符解析AnalyzeStr();//4.处理内建命令if(DoBuildin())continue;//5.fork() 创建子进程,execvp() 替换子进程RunCommand();}return 0;
}

相关文章:

模拟shell小程序

接下来利用我们当前的知识&#xff0c;撰写一个简单的shell外壳程序。 1.shell原理 shell的原理是实际上就是运行了一个父进程&#xff0c;然后创建出子进程&#xff0c;最后使用进程替换调用&#xff0c;替换成其他程序。 2.shell实现 2.1.死循环 首先一个shell一旦运行起…...

webpack配置全局scss

webpack配置全局scss 效果&#xff1a;a.vue使用index.scss中定义的$mainWidth就无需 import "xxxxxxx/index.scss"文件 src/assets/styles/index.scss $mainWidth: 1280px; $red: red src/views/a.vue .aaa {color: $red; } vue.config.js module.exports {…...

想面试前端工程师,必须掌握哪些知识和技能?【云驻共创】

在当今的数字化时代&#xff0c;前端工程师扮演着至关重要的角色。他们负责设计和开发用户界面&#xff0c;使得用户能够与应用程序或网站进行互动。为了找到最出色的前端工程师&#xff0c;你需要了解哪些技能和知识是必备的&#xff0c;同时也要掌握一些面试技巧和常见的面试…...

京东数据分析(京东数据采集):2023年10月京东平板电视行业品牌销售排行榜

鲸参谋监测的京东平台10月份平板电视市场销售数据已出炉&#xff01; 根据鲸参谋电商数据分析平台的相关数据显示&#xff0c;10月份&#xff0c;京东平台上平板电视的销量将近77万&#xff0c;环比增长约23%&#xff0c;同比则下降约30%&#xff1b;销售额为21亿&#xff0c;环…...

在 Linux 中,可以使用分号 (;) 或者 运算符来执行多条命令

在 Linux 中&#xff0c;你可以使用分号 (;) 或者 && 运算符来执行多条命令。 使用分号 (;) 分隔多条命令&#xff1a; command1 ; command2 这样会依次执行 command1 和 command2&#xff0c;不管前面的命令是否成功。 使用 && 运算符分隔多条命令&#xff1…...

一些必备的 Redis 命令 | Navicat

Redis 是一种快速的内存数据结构存储系统&#xff0c;因其处理键值对的能力而备受推崇。在本文&#xff0c;我们将探索一些不可或缺的 Redis 命令&#xff08;不包括之前介绍过的涉及键的命令&#xff09;&#xff0c;解锁这个强大工具的真正潜力。同时&#xff0c;我们也将了解…...

神经网络常用激活函数详解

&#x1f380;个人主页&#xff1a; https://zhangxiaoshu.blog.csdn.net &#x1f4e2;欢迎大家&#xff1a;关注&#x1f50d;点赞&#x1f44d;评论&#x1f4dd;收藏⭐️&#xff0c;如有错误敬请指正! &#x1f495;未来很长&#xff0c;值得我们全力奔赴更美好的生活&…...

UVA11584划分成回文串 Partitioning by Palindromes

划分成回文串 Partitioning by Palindromes 题面翻译 回文子串(palind) 问题描述&#xff1a; 当一个字符串正序和反序是完全相同时&#xff0c;我们称之为“回文串”。例如“racecar”就是一个回文串&#xff0c;而“fastcar”就不是。现在给一个字符串s&#xff0c;把它分…...

第十一章 将对象映射到 XML - 控制流属性的映射形式

文章目录 第十一章 将对象映射到 XML - 控制流属性的映射形式控制流属性的映射形式控制预计属性的可用性禁用映射%XML.Adapter 中的方法 第十一章 将对象映射到 XML - 控制流属性的映射形式 控制流属性的映射形式 对于流属性&#xff0c;XMLPROJECTION 的选项如下&#xff1a…...

torchvision中的标准ResNet50网络结构

注&#xff1a;仅用以记录学习 打印出来的网络结构如下&#xff1a; from torchvision import models model models.resnet50(pretrainedFalse) print("model: ", model) 结构&#xff1a; ResNet((conv1): Conv2d(3, 64, kernel_size(7, 7), stride(2, 2), padd…...

Java 多线程之 synchronized (互拆锁/排他锁/非观锁)

文章目录 一、概述二、使用方法三、测试示例 一、概述 在Java中&#xff0c;synchronized 关键字用于实现线程之间的同步。提供了一种简单而强大的机制来控制多个线程之间的并发访问&#xff0c;确保共享资源的安全性和一致性。它解决了多线程环境中的竞态条件、数据竞争和内存…...

开源vs闭源大模型如何塑造技术的未来?开源模型的优劣势未来发展方向

开源vs闭源大模型如何塑造技术的未来&#xff1f;开源模型的优劣势&未来发展方向 写在最前面一、开源与闭源&#xff1a;定义与历史背景开源和闭源的定义开源大模型&#xff1a;社区驱动的创新 二、开源和闭源的优劣势比较开源大模型&#xff08;瓶颈&#xff09;数据&…...

如何使用无代码系统搭建软件平台?有哪些开源无代码开发平台?

无代码是什么 无代码开发&#xff0c;也称为零代码&#xff08;Zero Code&#xff09;开发&#xff0c;是一种技术概念。无代码开发无需代码基础&#xff0c;适合业务人员、IT开发及其他各类人员使用。他们通过无代码开发平台快速构建应用&#xff0c;并适应各种需求变化&#…...

微信怎么设置自动回复?

自动回复的用处 微信自动回复可以提高沟通效率。当你无法立即回复消息时&#xff0c;设置自动回复可以让对方知道你的情况&#xff0c;并且不会因为长时间没有回复而产生误解或不满。 微信自动回复可以节省时间和精力。如果你经常收到类似的询问或回复&#xff0c;通过设置自动…...

基于Vue3的低代码开发平台——JNPF

目录 一、什么是Vue.js &#xff1f; 二、Jnpf-Web-Vue3 的技术栈介绍 &#xff08;1&#xff09;Vue3.x &#xff08;2&#xff09;Vue-router4.x &#xff08;3&#xff09;Vite4.x &#xff08;4&#xff09;Ant-Design-Vue3.x &#xff08;5&#xff09;TypeScript &#x…...

Thinkphp6 模型 指定字段自增的方法

tp6要使用Db类必须使用门面方式&#xff08;think\facade\Db&#xff09;调用。 use think\facade\Db; 然后&#xff0c;用Db::raw就可以实现指定字段自增了。...

WhatsApp开发客户攻略来袭!还有你不知道的账号解封秘籍!

别人用 WhatsApp 都是订单多到爆单&#xff0c;自己用 WhatsApp 却是订单、客户寥寥无几甚至账号被封&#xff1f;想必外贸从业者在用 WhatsApp 开发客户的时候都有这样的烦恼&#xff0c;今天这篇文章就和大家聊一聊怎么用 WhatsApp 高效地开发客户。 WhatsApp 开发客户的优势…...

Linux C 基于tcp多线程在线聊天室

多线程在线聊天室 概述客户端服务端 概述 客户端实现了判单用户登录结果、防止单回车字符发送、保存和显示历史聊天记录&#xff08;仅自己&#xff09;、退出聊天室功能。   服务端实现了验证用户是否已经存在&#xff08;支持最大64用户连接&#xff09;支持广播用户进入退…...

代码随想录算法训练营第23期day60|84.柱状图中最大的矩形

一、84.柱状图中最大的矩形 力扣题目链接 42接雨水 是找每个柱子左右两边第一个大于该柱子高度的柱子&#xff0c;而本题是找每个柱子左右两边第一个小于该柱子的柱子。 本题是要找每个柱子左右两边第一个小于该柱子的柱子&#xff0c;所以从栈头&#xff08;元素从栈头弹出…...

vue动态获取目录结构进行配置静态路由

文章目录 前言定义项目页面格式一、vite 配置动态路由新建 /router/utils.ts引入 /router/utils.ts 二、webpack 配置动态路由总结如有启发&#xff0c;可点赞收藏哟~ 前言 项目中动态配置路由可以减少路由配置时间&#xff0c;并可减少配置路由出现的一些奇奇怪怪的问题 路由…...

OpenLayers 可视化之热力图

注&#xff1a;当前使用的是 ol 5.3.0 版本&#xff0c;天地图使用的key请到天地图官网申请&#xff0c;并替换为自己的key 热力图&#xff08;Heatmap&#xff09;又叫热点图&#xff0c;是一种通过特殊高亮显示事物密度分布、变化趋势的数据可视化技术。采用颜色的深浅来显示…...

Java - Mysql数据类型对应

Mysql数据类型java数据类型备注整型INT/INTEGERint / java.lang.Integer–BIGINTlong/java.lang.Long–––浮点型FLOATfloat/java.lang.FloatDOUBLEdouble/java.lang.Double–DECIMAL/NUMERICjava.math.BigDecimal字符串型CHARjava.lang.String固定长度字符串VARCHARjava.lang…...

代码随想录刷题day30

1、零钱兑换II 给你一个整数数组 coins 表示不同面额的硬币&#xff0c;另给一个整数 amount 表示总金额。 请你计算并返回可以凑成总金额的硬币组合数。如果任何硬币组合都无法凑出总金额&#xff0c;返回 0 。 假设每一种面额的硬币有无限个。 题目数据保证结果符合 32 位带…...

嵌入式学习笔记DAY33(网络编程——TCP)

一、网络架构 C/S &#xff08;client/server 客户端/服务器&#xff09;&#xff1a;由客户端和服务器端两个部分组成。客户端通常是用户使用的应用程序&#xff0c;负责提供用户界面和交互逻辑 &#xff0c;接收用户输入&#xff0c;向服务器发送请求&#xff0c;并展示服务…...

软件工程 期末复习

瀑布模型&#xff1a;计划 螺旋模型&#xff1a;风险低 原型模型: 用户反馈 喷泉模型:代码复用 高内聚 低耦合&#xff1a;模块内部功能紧密 模块之间依赖程度小 高内聚&#xff1a;指的是一个模块内部的功能应该紧密相关。换句话说&#xff0c;一个模块应当只实现单一的功能…...

Vue3 PC端 UI组件库我更推荐Naive UI

一、Vue3生态现状与UI库选择的重要性 随着Vue3的稳定发布和Composition API的广泛采用&#xff0c;前端开发者面临着UI组件库的重新选择。一个好的UI库不仅能提升开发效率&#xff0c;还能确保项目的长期可维护性。本文将对比三大主流Vue3 UI库&#xff08;Naive UI、Element …...

Copilot for Xcode (iOS的 AI辅助编程)

Copilot for Xcode 简介Copilot下载与安装 体验环境要求下载最新的安装包安装登录系统权限设置 AI辅助编程生成注释代码补全简单需求代码生成辅助编程行间代码生成注释联想 代码生成 总结 简介 尝试使用了Copilot&#xff0c;它能根据上下文补全代码&#xff0c;快速生成常用…...

Docker 镜像上传到 AWS ECR:从构建到推送的全流程

一、在 EC2 实例中安装 Docker&#xff08;适用于 Amazon Linux 2&#xff09; 步骤 1&#xff1a;连接到 EC2 实例 ssh -i your-key.pem ec2-useryour-ec2-public-ip步骤 2&#xff1a;安装 Docker sudo yum update -y sudo amazon-linux-extras enable docker sudo yum in…...

在 Vue 的template中使用 Pug 的完整教程

在 Vue 的template中使用 Pug 的完整教程 引言 什么是 Pug&#xff1f; Pug&#xff08;原名 Jade&#xff09;是一种高效的网页模板引擎&#xff0c;通过缩进式语法和简洁的写法减少 HTML 的冗长代码。Pug 省略了尖括号和闭合标签&#xff0c;使用缩进定义结构&#xff0c;…...

unipp---HarmonyOS 应用开发实战

HarmonyOS 应用开发实战指南 1. 开篇&#xff1a;为什么选择 HarmonyOS&#xff1f; 最近在开发鸿蒙应用时&#xff0c;发现很多开发者都在问&#xff1a;为什么要选择 HarmonyOS&#xff1f;这里分享一下我的看法&#xff1a; 生态优势 华为手机用户基数大&#xff0c;市场潜…...