实用指南:苍茫命令行:linux模拟实现,书写微型bash

news/2025/10/5 8:07:40/文章来源:https://www.cnblogs.com/ljbguanli/p/19126220

实用指南:苍茫命令行:linux模拟实现,书写微型bash

?前言

Linux 系统主要分为内核(kernel)和 外壳(shell),普通用户是无法接触到内核的,因此实际在进行操作时是在和外壳程序打交道,在 shell 外壳之上存在 命令行解释器(bash),负责接收并执行用户输入的指令,本文模拟实现的就是一个 简易版命令行解释器

在这里插入图片描述
?️正文
1、bash本质
在模拟实现前,先得了解 bash 的本质

bash 也是一个进程,并且是不断运行中的进程
证明:常显示的命令输入提示符就是 bash 不断打印输出的结果
在这里插入图片描述

输入指令后,bash 会创建子进程,并进行程序替换

bash 就是一个运行中的进程,因为进程间具有独立性,因此可以同时存在多个 bash,这也是多用户登录 Linux 可以同时使用 bash 的重要原因

2、需求分析

bash 需要帮我们完成命令解释+程序替换的任务,因此它至少要具备以下功能:

进程相关知识都已经在前面介绍过了,本文着重介绍的是其他步骤及细节

3、基本框架

抛开指令接收、切割、替换时的细节,简易版 bash 代码基本框架如下:

#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <sys/wait.h>#include <sys/types.h>#include <string.h>#include <assert.h>//指令分割函数void split(char* argv[ARGV_SIZE], char* ps){}int main(){//这是一个始终运行的程序:bashwhile(1){//打印提示符printf("[User@myBash default]$ "); //可以自定义,跟着标准走fflush(stdout); //手动清空缓冲区//读取指令//指令分割//子进程进行程序替换pid_t id = fork();if(id == 0){//直接执行程序替换,这里使用 execvpexecvp(); //具体细节先忽略exit(168); //替换失败后返回,这个值可以自定义 [0,255]}//父进程等待子进程终止,回收僵尸进程int status = 0;waitpid(id, &status, 0); //在等待队列中阻塞if(WIFEXITED(status)){//假如程序替换失败//关于打印的错误信息:也可以自定义,格式跟着标准走if(WEXITSTATUS(status) == 168)printf("%s: Error - %s\n", argv[0],"The directive is not yet defined");}else //如果子进程被异常终止,打印相关信息printf("process run fail! [code_dump]:%d [exit_signal]:%d\n",(status >>7) &1, status & 0x7F); //子进程异常终止的情况}return 0;}

这只是简易版 bash 的基本框架,其他细节将会在后续补充完整

4、核心内容

核心内容主要为 读取、切割、替换 这三部分,逐一实现,首先从指令读取开始

在这里插入图片描述
4.1、指令读取
读取指令前,首先要清楚待读取命令可能有多长

#
define COM_SIZE 1024
char command[COM_SIZE]
;
//缓冲区

得到缓冲区后,就得考虑什么是指令?如何读取指令?

  • Linux 中的大部分指令由 指令 [选项] 构成,在 指令 [选择] 间有空格
  • 常规的 scanf 无法正常读取指令,因为空格会触发输入缓冲区刷新
  • 这里主要使用fgets逐行读取,可以读取到空格
//读取指令
//因为有空格,所以需要逐行读取
fgets(command, COM_SIZE, stdin
)
;
assert(command)
;
//不能输入空指令
(
void
)command;
//防止在 Release 版本中出错
command[strlen(command) - 1] = '\0'
;
//将最后一个字符 \n 变成 \0

注意: 可能存在读取失败的情况,assert 断言解决;因为 fgets 也会把最后的 ‘\n’ 读进去,为了避免出错,手动置为 ‘\0’

4.2、指令分割

获得指令后,就需要将指令进行分割

为何要分割指令?

如何分割指令?

指令分割后呢?

#
define ARGV_SIZE 64
//指令分割
//将连续的指令分割为 argv 表
char* argv[ARGV_SIZE]
;
//指针数组
split(argv, command)
;

利用strtok实现指令分割函数 split()

#
define DEF_CHAR " " //预设分割项,需为字符串
void split(
char* argv[ARGV_SIZE]
,
char* ps)
{
assert(argv && ps)
;
//调用 C语言 中的 strtok 函数分割字符串
int pos = 0
;
argv[pos++] = strtok(ps, DEF_CHAR)
;
//有空格就分割
while(argv[pos++] = strtok(NULL
, DEF_CHAR)
)
;
//不断分割
argv[pos] = NULL
;
//确保安全
}

注意: 指令分割结束后,需要在添加 argv 表结尾 NULL

4.3、程序替换

获得实际可用的 argv 表后,就可以开始子进程程序替换操作了

这里使用的是函数 execvp,理由:

//子进程进行程序替换
pid_t id = fork(
)
;
if(id == 0
)
{
//直接执行程序替换,这里使用 execvp
execvp(argv[0]
, argv)
;
exit(168
)
;
//替换失败后返回
}

注意: 程序替换成功后,exit(168) 语句不会执行

5、特殊情况处理

对特殊情况进行处理,使 myBash 更加完善

5.1、ls 显示高亮
系统中的 bash 在面对ls等文件显示指令时,不仅会显示内容,还会将特殊文件做颜色高亮处理,比如在我的环境下,可执行文件显示为绿色

实现原理

处理这个问题很简单,在指令分割结束后,判断是否为 ls,如果是,就在argv表后尾插入语句--color=auto即可

//特殊处理
//颜色高亮处理,识别是否为 ls 指令
if(strcmp(argv[0]
, "ls"
) == 0
)
{
int pos = 0
;
while(argv[pos++]
)
;
//找到尾
argv[pos - 1] = (
char*
)"--color=auto"
;
//添加此字段
argv[pos] = NULL
;
//结新尾
}

注意:

  • 因为 argv 表中的元素类型为 char*,所以在尾插语句时,需要进行类型转换
  • 尾插语句后,需要再次添加结尾,确保安全

5.2、内建命令

内建命令是比较特殊的命令,不同于普通命令直接进行程序替换,内建命令需要进行特殊处理,比如 cd 命令调用系统级接口 chdir 让 父进程(myBash) 进行目录间的移动
在这里插入图片描述

5.3、cd

首先实现不同目录间的切换

切换的本质:令当前 bash 移动至另一个目录下,不能直接使用 子进程 ,因为需要移动的是 父进程(bash)

对于当前的 myBash 来说,cd 没有丝毫效果,因为此时 指令会被拆分后交给子进程处理,这个方向本身就是错误的

特殊情况特殊处理,同ls高亮一样,对指令进行识别,如果识别到cd命令,就直接调用chdir函数令当前进程myBash移动至指定目录即可(不必再创建子进程进行替换)

//目录间移动处理
if(strcmp(argv[0]
, "cd"
) == 0
)
{
//直接调用接口,然后 continue 不再执行后续代码
if(strcmp(argv[1]
, "~"
) == 0
)
chdir("/home"
)
;
//回到家目录
else
if(strcmp(argv[1]
, "-"
) == 0
)
chdir(getenv("OLDPWD"
)
)
;
else
if(argv[1]
)
chdir(argv[1]
)
;
//argv[1] 中就是路径
continue
;
//终止此次循环
}

特殊情况特殊处理,同 ls 高亮一样,对指令进行识别,如果识别到 cd 命令,就直接调用chdir函数令当前进程myBash 移动至指定目录即可(不必再创建子进程进行替换)

//目录间移动处理
if(strcmp(argv[0]
, "cd"
) == 0
)
{
//直接调用接口,然后 continue 不再执行后续代码
if(strcmp(argv[1]
, "~"
) == 0
)
chdir("/home"
)
;
//回到家目录
else
if(strcmp(argv[1]
, "-"
) == 0
)
chdir(getenv("OLDPWD"
)
)
;
else
if(argv[1]
)
chdir(argv[1]
)
;
//argv[1] 中就是路径
continue
;
//终止此次循环
}

注意:

  • 如果路径为空,不进行操作;
  • 如果路径为 ~,回到家目录;
  • cd - 指令依赖于 OLDPWD 这个环境变量,直接拿来用即可

5.4、export

export 添加环境变量,添加的是父进程 myBash 的环境变量,而非子进程,需要特殊处理

解决方法:

为何不能直接通过 putenv 添加至环境变量表中?

错误体现:直接使用 putenv(argv[1]),导致第一次添加可能成功,但第二次添加后,第一次的环境变量会被覆盖

正确解法是借助缓冲区 myEnv

#
define COM_SIZE 1024
#
define ARGV_SIZE 64
char myEnv[ARGV_SIZE][COM_SIZE]
;
//二维数组
int env_pos = 0
;
//专门维护此缓冲区

注意: 此缓冲区定义在循环之外

char myEnv[COM_SIZE][ARGV_SIZE]
;
//大小与前面有关
int env_pos = 0
;
//专门维护缓冲区
//这是一个始终运行的程序:bash
while(1
)
{
//…… 省略部分代码
//环境变量相关
if(strcmp(argv[0]
, "export"
) == 0
)
{
if(argv[1]
)
{
strcpy(myEnv[env_pos]
, argv[1]
)
;
putenv(myEnv[env_pos++]
)
;
}
continue
;
//一样需要提前结束循环
}
}

除了export需要特殊处理外,env 查看环境变量表也需要特殊处理,因为此时的 env 查看的是 父进程(myBash) 的环境变量表,因此不需要将指令交给 子进程 处理

//注意:此函数实现于主函数外
void showEnv(
)
{
extern
char** environ;
//使用当前进行的环境变量表
int pos = 0
;
for(
; environ[pos]
;
printf("%s\n"
, environ[pos++]
)
)
;
}
//环境变量表
if(strcmp(argv[0]
, "env"
) == 0
)
{
showEnv(
)
;
//调用函数,打印父进程的环境变量表
continue
;
//提前结束本次循环
}

完善后,env 指令显示的才是正确进程的环境变量表

5.5、echo

echo 命令也属于内建命令,其能实现很多功能,比如:查看环境变量、查看最近一个进程的退出码、输出重定向等,其中前两个实现比较简单,最后一个需要 基础IO 相关知识,后续更新补上

查看环境变量

echo 指令查看环境变量时,指令为 echo $ 环境变量,可以先判断 argv[1][0] 是否为 $,如果是,就直接根据 argv[1][1] 获取环境变量信息并打印即可

代码实现如下

//echo 相关
//只有 echo $ 才做特殊处理(环境变量+退出码)
if(strcmp(argv[0]
, "echo"
) == 0 && argv[1][0] == '$'
)
{
if(argv[1] && argv[1][0] == '$'
)
printf("%s\n"
, getenv(argv[1] + 1
)
)
;
continue
;
}

echo 还能查看退出码:echo $?,对上述程序进行改造即可实现

退出码从何而来?

  • 很简单,父进程在等待子进程结束后,可以轻而易举的获取其退出码
  • 将退出码保存在一个全局变量中,供echo $?指令使用即可
int exit_code = 0
;
//保存退出码的全局变量

代码实现:

//echo 相关
//只有 echo $ 才做特殊处理(环境变量+退出码)
if(strcmp(argv[0],
"echo"
) == 0 && argv[1][0] == '$'
)
{
if(argv[1] && argv[1][0] == '$'
)
{
if(argv[1][1] == '?'
)
printf("%d\n", exit_code)
;
else
printf("%s\n", getenv(argv[1] + 1
))
;
}
continue
;
}

5.6、重定向

重定向的本质:关闭默认输出/输入流,打开新的文件流,从其中写入/读取数据

重定向的三种情况:

所以实现重定向的关键在于判断指令中是否含有 >>>< 这三个字符,如果有,就具体问题具体分析,完成重定向

具体实现步骤:

open 函数的打开选项

O_RDONLY //只读
O_WRONLY | O_CREAT | O_TRUNC //只写
O_WRONLY | O_CREAT | O_APPEND //追加

标准流交换函数 dup2

//给参数1传打开文件后的文件描述符,给参数2传递待关闭的标准流
//读取:关闭0号流
//写入、追加:关闭1号流
int dup2(
int oldfd,
int newfd)
;

下面是具体代码实现

//在读取指令后,就进行判断:是否需要重定向
//重定向
//在获取指令后进行判断
//如果成立,则获取目标文件名 filename
char *filename = checkDir(command)
;
//枚举类型,用于判断不同的文件打开方式
enum redir
{
REDIR_INPUT = 0
, //读取
REDIR_OUTPUT, //写入
REDIR_APPEND, //追加
REDIR_NONE //空
}redir_type = REDIR_NONE;
//创建对象 redir_type,默认为 NONE
//检查是否出现重定向符
char* checkDir(
char* command)
{
//从右往左遍历,遇到 > >> < 就置为 '\0'
size_t end = strlen(command)
;
//与返回值相匹配
char* ps = command + end;
//为了避免出现无符号-1,这里采取错位的方法
while(end != 0
)
{
if(command[end - 1] == '>'
)
{
if(command[end - 2] == '>'
)
{
command[end - 2] = '\0'
;
redir_type = REDIR_APPEND;
return ps;
}
command[end - 1] = '\0'
;
redir_type = REDIR_OUTPUT;
return ps;
}
else
if(command[end - 1] == '<'
)
{
command[end - 1] = '\0'
;
redir_type = REDIR_INPUT;
return ps;
}
//如果不是空格,就可以更新 ps指向
if(*(command + end - 1
) != ' '
)
ps = command + end - 1
;
end--
;
}
return NULL
;
//如果没有重定向符,就返回空
}
//子进程进行程序替换
pid_t id = fork(
)
;
if(id == 0
)
{
//判断是否需要进行重定向
if(redir_type == REDIR_INPUT)
{
int fd = open(filename, O_RDONLY)
;
dup2(fd, 0
)
;
//更改输入,读取文件 filename
}
else
if(redir_type == REDIR_OUTPUT)
{
int fd = open(filename, O_WRONLY | O_CREAT | O_TRUNC, 0666
)
;
dup2(fd, 1
)
;
//写入
}
else
if(redir_type == REDIR_APPEND)
{
int fd = open(filename, O_WRONLY | O_CREAT | O_APPEND, 0666
)
;
dup2(fd, 1
)
;
//追加
}
//直接执行程序替换,这里使用 execvp
execvp(argv[0]
, argv)
;
exit(168
)
;
//替换失败后返回
}

具体效果(A.txt 为空,B.txt 已存在内容,程序 a.out 可以读取字符串并输出):
在这里插入图片描述
注意: 当前实现的重定向只是最简单的标准流替换,实际重定向更加复杂

6、源码

本次实现的myBash如下所示,拷贝编译运行后,即可使用

#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <sys/wait.h>#include <sys/types.h>#include <string.h>#include <assert.h>#include <sys/stat.h>#include <fcntl.h>#define COM_SIZE 1024#define ARGV_SIZE 64#define DEF_CHAR " "void split(char* argv[ARGV_SIZE], char* ps){assert(argv &&ps);//调用 C语言 中的 strtok 函数分割字符串int pos = 0;argv[pos++] = strtok(ps, DEF_CHAR); //有空格就分割while(argv[pos++] = strtok(NULL, DEF_CHAR)); //不断分割argv[pos] = NULL; //确保安全}void showEnv(){extern char** environ; //使用当前进行的环境变量表int pos = 0;for(; environ[pos]; printf("%s\n", environ[pos++]));}//枚举类型,用于判断不同的文件打开方式enum redir{REDIR_INPUT = 0,REDIR_OUTPUT,REDIR_APPEND,REDIR_NONE}redir_type = REDIR_NONE; //创建对象 redir_type,默认为 NONE//检查是否出现重定向符char* checkDir(char* command){//从右往左遍历,遇到 >>>< 就置为 '\0'size_t end = strlen(command); //与返回值相匹配char* ps = command + end; //为了避免出现无符号-1,这里采取错位的方法while(end != 0){if(command[end - 1] == '>'){if(command[end - 2] == '>'){command[end - 2] = '\0';redir_type = REDIR_APPEND;return ps;}command[end - 1] = '\0';redir_type = REDIR_OUTPUT;return ps;}else if(command[end - 1] == '<'){command[end - 1] = '\0';redir_type = REDIR_INPUT;return ps;}//如果不是空格,就可以更新 ps指向if(*(command + end - 1) != ' ')ps = command + end - 1;end--;}return NULL; //如果没有重定向符,就返回空}int main(){char myEnv[COM_SIZE][ARGV_SIZE]; //大小与前面有关int env_pos = 0; //专门维护缓冲区int exit_code = 0; //保存退出码的全局变量//这是一个始终运行的程序:bashwhile(1){char command[COM_SIZE]; //存放指令的数组(缓冲区)//打印提示符printf("[User@myBash default]$ ");fflush(stdout);//读取指令//因为有空格,所以需要逐行读取fgets(command, COM_SIZE, stdin);assert(command); //不能输入空指令(void)command; //防止在 Release 版本中出错command[strlen(command) - 1] = '\0'; //将最后一个字符 \n 变成 \0//重定向//在获取指令后进行判断//如果成立,则获取目标文件名 filenamechar *filename = checkDir(command);//指令分割//将连续的指令分割为 argv 表char* argv[ARGV_SIZE];split(argv, command);//特殊处理//颜色高亮处理,识别是否为 ls 指令if(strcmp(argv[0],"ls") == 0){int pos = 0;while(argv[pos++]); //找到尾argv[pos - 1] = (char*)"--color=auto"; //添加此字段argv[pos] = NULL; //结尾}//目录间移动处理if(strcmp(argv[0],"cd") == 0){//直接调用接口,然后 continue 不再执行后续代码if(strcmp(argv[1],"~") == 0)chdir("/home"); //回到家目录else if(strcmp(argv[1],"-") == 0)chdir(getenv("OLDPWD"));else if(argv[1])chdir(argv[1]); //argv[1] 中就是路径continue; //终止此次循环}//环境变量相关if(strcmp(argv[0],"export") == 0){if(argv[1]){strcpy(myEnv[env_pos], argv[1]);putenv(myEnv[env_pos++]);}continue; //一样需要提前结束循环}//环境变量表if(strcmp(argv[0],"env") == 0){showEnv(); //调用函数,打印父进程的环境变量表continue; //提前结束本次循环}//echo 相关//只有 echo $ 才做特殊处理(环境变量+退出码)if(strcmp(argv[0],"echo") == 0 && argv[1][0] == '$'){if(argv[1] && argv[1][0] == '$'){if(argv[1][1] == '?')printf("%d\n", exit_code);elseprintf("%s\n", getenv(argv[1] + 1));}continue;}//子进程进行程序替换pid_t id = fork();if(id == 0){//判断是否需要进行重定向if(redir_type == REDIR_INPUT){int fd = open(filename, O_RDONLY);dup2(fd, 0); //更改输入,读取文件 filename}else if(redir_type == REDIR_OUTPUT){int fd = open(filename, O_WRONLY | O_CREAT | O_TRUNC, 0666);dup2(fd, 1); //写入}else if(redir_type == REDIR_APPEND){int fd = open(filename, O_WRONLY | O_CREAT | O_APPEND, 0666);dup2(fd, 1); //追加}//直接执行程序替换,这里使用 execvpexecvp(argv[0], argv);exit(168); //替换失败后返回}//父进程等待子进程终止int status = 0;waitpid(id, &status, 0); //在等待队列中阻塞exit_code = WEXITSTATUS(status);if(WIFEXITED(status)){//假如程序替换失败if(exit_code == 168)printf("%s: Error - %s\n", argv[0],"The directive is not yet defined");}elseprintf("process run fail! [code_dump]:%d [exit_signal]:%d\n",(status >>7) &1, status & 0x7F); //子进程异常终止的情况}return 0;}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/928017.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

营销型网站策划建设wordpress 转换 织梦

百科词条创建上去是相当不易的&#xff0c;同时修改也是如此&#xff0c;一般情况下&#xff0c;百科词条是不需要修改的&#xff0c;但是很多时候企业或是人物在近期收获了更多成就或是有更多的变动&#xff0c;这个时候就需要补充维护词条了&#xff0c;如何修改百科内容&…

微信餐饮微网站建设重庆网站建立

Kafka 1.基于Pull的模式来处理消息消费 2.追求高吞吐量 3.一开始的目的就是日志收集和传输 4.0.8版本开始支持复制&#xff0c;不支持事务&#xff0c;对消息的重复、丢失、错误没有严格要求、适合产生大量数据的互联网服务的数据收集业务. RabbitMQ RabbitMQ是使用Erlang语…

品牌网站建设9小蝌蚪9a广州天河区做网站

视频监控技术从传统监控到智能化升级的过程是一个技术革新和应用场景拓展的过程。智能视频监控系统通过集成AI和机器学习算法&#xff0c;能够实现行为分析、人脸识别和异常事件检测等功能&#xff0c;提升了监控的准确性和响应速度。这些系统不仅用于传统的安全防护&#xff0…

公司网站建设费怎么入账deppt模板网

一、简介 股票上涨和下跌,创造出像海浪一样难以预测的模式和走势。然而,就像科学家通过了解下面的水流来预测波浪的运动一样,我们也可以使用类似的工具破译股票市场的一些模式。 通过利用小波变换的力量,我们深入表面,试图揭示驱动股价的深层原因。这段旅程不仅仅涉及数字…

微信群公告如何做网站链接工商注册流程和需要的资料

WP中获取POST有两个主要函数&#xff0c;Get_post()和Get_Posts()。一个是获取单文章&#xff0c;另外一个是获取多文章&#xff0c;其中&#xff0c;官网对Get_posts()函数的描述很简单。但有的时候描述越简单的函数&#xff0c;使用起来却并没有那么简单。下面我通过一个案例…

CF2149题解

A. Be Positivecode #include<bits/stdc++.h> using namespace std; const int NN = 1e4; int T; int n,a[NN]; int cnt1,cnt2; void solve(){cnt1 = cnt2 = 0;cin >> n;for(int i = 1; i <= n; ++i){c…

教育网站建设改版湘潭做网站 z磐石网络

网站爬虫&#xff08;Web Scrapers&#xff09;是一种自动化工具&#xff0c;用于抓取网页上的数据。虽然合法的搜索引擎爬虫有助于提升网站的可见度&#xff0c;但恶意爬虫可能带来数据盗窃、版权侵犯等问题&#xff0c;甚至造成服务器过载。本文将探讨如何有效预防网站受到恶…

网站建设 企业文化东莞南城网站开发公司

using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System

经营地址怎么在国税网站做更改网线水晶头制作过程

最近在看深入理解计算机系统这本书&#xff0c;上面提到了在32位机器和64机器中int类型都占用4个字节。后来&#xff0c;查了The C Programming language这本书&#xff0c;里面有一句话是这样的&#xff1a;Each compiler is free to choose appropriate sizes for its own ha…

咸阳做网站的公司网络有限公司简介

使用命令下载&#xff1a;参考&#xff1a;解决nvidia-smi报错&#xff1a;NVIDIA-SMI has failed because it couldn‘t communicate with the NVIDIA driver.-CSDN博客

2025 年压滤机厂家最新推荐排行榜:隔膜压滤机,污泥压滤机,真空压滤机,板框压滤机,带式压滤机优质企业权威评选及选购指南

在工业固液分离领域持续升级的当下,压滤机作为核心设备,已深度渗透化工、矿山、环保等多元场景,隔膜、污泥、真空等细分类型设备的性能需求愈发差异化。然而当前市场品牌繁杂,产品在过滤精度、能耗控制、适配性等方…

浙江网站开发网络营销方式举例

文章目录一、环境准备1. 安装node2. 配置镜像二、安装Vue CLI2.1. 查看当前vuecli版本2.2. 安装最新版本2.3. 安装指定版本三、创建web项目3.1. 指定创建的项目名称3.2. 安装序列图3.3. 安装序列图简述一、环境准备 声明&#xff1a;命令均在在cmd窗口执行 1. 安装node 2. 配置…

嘉定专业网站建设企业网站源码去一品资源网

之前写了《Linux配置IP-SAN&#xff08;iSCSI&#xff09;》&#xff0c;现在简单记录Windows配置IP-SAN&#xff08;iSCSI&#xff09;&#xff0c;基本过程都是一样的。一些原理请参考《Linux配置IP-SAN&#xff08;iSCSI&#xff09;》&#xff0c;更详细一些。 目录 一、确…

2025 年搅拌器厂家最新推荐排行榜:涵盖立式、不锈钢、侧入式等多类型设备,深度解析实力厂商

当前搅拌器市场厂商众多,产品类型繁杂,质量与性能参差不齐,企业在采购时常面临选型难、辨优劣的困境。部分产品存在材料劣质、效率低下等问题,不仅影响生产进度,还可能增加成本损耗。为帮助企业精准筛选优质搅拌器…

东营网站建设推广哪家好湖南住房和城乡建设网门户网站

本章重点 为什么存在动态内存分配 动态内存函数的介绍 malloc free calloc realloc 常见的动态内存错误 几个经典的笔试题 柔性数组 1. 为什么存在动态内存分配 我们已经掌握的内存开辟方式有&#xff1a; int val 20;//在栈空间上开辟四个字节 char arr[10] {0}…

越秀营销型网站建设wordpress博客模板seo

使用Vue作为前端开发技术栈的同学&#xff0c;在使用Vue时都会有一些好奇&#xff1a;为啥我们的响应式变量要在data中定义&#xff1f;Vue是如何监听到变化&#xff0c;实现响应式的&#xff1f;这次我们就来探究一下&#xff0c;Vue2的响应式原理。 对象的响应式 修改属性描…

免费dede企业网站模板做企业平台的网站有哪些方面

考试周连考不复习就挂科了 一直没更新十分抱歉 今天开始在周日前补回来 491.递增子序列 在90.子集I中我们是通过排序&#xff0c;再加一个标记数组来达到去重的目的。 而本题求自增子序列&#xff0c;是不能对原数组进行排序的&#xff0c;排完序的数组都是自增子序列了。 …

电子商务网站开发步骤西安家政公司网站建设

一、安装步骤1、使用whichgcc命令发现gcc没有安装2、拷贝gcc-3.2.2-5.i386.rpm 尝试安装说明在这之前还需要先装binutils、cpp、glibc-devel这三个包3、拷贝binutils-2.13.90.0.18-9.i386.rpmcpp-3.2.2-5.i386.rpmglibc-devel-2.3.2-11.9.i386.rpm 到install目录分别安装rpm -i…

网站界面风格设计描述网站开发的实验心德

工厂方法模式 1&#xff09;问题 简单工厂模式 当需要引入新产品时&#xff0c;由于静态工厂方法通过所传入参数的不同来创建不同的产品&#xff0c;需要修改工厂类的源代码。 2&#xff09;概述 针对不同的产品提供不同的工厂&#xff0c;系统提供一个与产品等级结构对应…

2025 年最新推荐承烧板厂家排行榜:筛选优质企业,破解采购难题,赋能高温工业生产

当前,高温工业领域对承烧板的需求持续攀升,然而市场上承烧板制造商良莠不齐,部分企业为降低成本,在原材料选用和生产工艺上敷衍了事,导致产品在高温环境下易出现开裂、变形等问题,严重影响生产效率,还带来安全隐…