0


【Linux】模拟实现命令行解释器shell

shell本质就是一个进程,它提供了一个用户界面,用于接收用户输入的命令,并将这些命令解释成操作系统能够理解和执行的操作。它充当了用户和操作系统内核之间的中介。例如,在 Linux 系统中,当用户在终端输入

  1. ls

命令时,shell 会解释这个命令,告诉操作系统去列出当前目录下的文件和目录信息。

下面是模拟实现shell的基本代码:

  1. #include <iostream>
  2. #include <string>
  3. #include <cstring>
  4. #include <cstdio>
  5. #include <cstdlib>
  6. #include <unistd.h>
  7. #include <sys/types.h>
  8. #include <sys/wait.h>
  9. using namespace std;
  10. const int basesize = 1024;
  11. const int argvnum = 64;
  12. const int envnum = 64;
  13. //全局的命令行参数表
  14. char *gargv[argvnum];
  15. int gargc = 0;
  16. //我自己的环境变量
  17. char *genv[envnum];
  18. //全局变量,用来表示退出结果
  19. int lastcode = 0;
  20. // 全局的工作路径
  21. char pwd[basesize];
  22. char pwdenv[basesize * 2];
  23. string GetName()
  24. {
  25. string name = getenv("USER");
  26. return name.empty() ? "None" : name;
  27. }
  28. string GetHostName()
  29. {
  30. char hostname[basesize];
  31. gethostname(hostname, sizeof(hostname));
  32. ;
  33. return gethostname(hostname, sizeof(hostname)) != 0 ? "None" : hostname;
  34. }
  35. string GetPwd()
  36. {
  37. // getcwd获取当前工作路径
  38. if (nullptr == getcwd(pwd, sizeof(pwd)))
  39. return "None";
  40. // 讲获取的当前路径输入到pwdenv
  41. snprintf(pwdenv, sizeof(pwdenv), "PWD=%s", pwd);
  42. // 导入环境变量
  43. putenv(pwdenv);
  44. return pwd;
  45. // string pwd = getenv("PWD");
  46. // return pwd.empty() ? "None" : pwd;
  47. }
  48. string LastDir()
  49. {
  50. string curr = GetPwd();
  51. if(curr == "/" || curr == "None") return curr;
  52. // /home/xzl/xxx
  53. size_t pos = curr.rfind("/");
  54. if(pos == std::string::npos) return curr;
  55. return curr.substr(pos+1);
  56. }
  57. string MakeCommandLine()
  58. {
  59. char Command_Line[basesize];
  60. snprintf(Command_Line, basesize, "[%s@%s %s]#",\
  61. GetName().c_str(), GetHostName().c_str(), LastDir().c_str());
  62. return Command_Line;
  63. }
  64. // 打印命令行提示符
  65. void PrintCommandLine()
  66. {
  67. printf("%s", MakeCommandLine().c_str());
  68. fflush(stdout);
  69. }
  70. // 获取用户命令
  71. bool GetCommandLine(char command_buffer[])
  72. {
  73. // 获取字符串
  74. char *result = fgets(command_buffer, basesize, stdin);
  75. if (!result)
  76. {
  77. return false;
  78. }
  79. command_buffer[strlen(command_buffer) - 1] = 0;
  80. if (strlen(command_buffer) == 0)
  81. return false;
  82. return true;
  83. }
  84. // 分析命令
  85. void ParseCommandLine(char command_buffer[])
  86. {
  87. memset(gargv, 0, sizeof(gargc));
  88. gargc = 0;
  89. const char *seq = " ";
  90. gargv[gargc++] = strtok(command_buffer, seq);
  91. while (gargv[gargc++] = strtok(nullptr, seq))
  92. ;
  93. gargc--;
  94. }
  95. void debug()
  96. {
  97. printf("argc: %d\n", gargc);
  98. for (int i = 0; gargv[i]; i++)
  99. {
  100. printf("argv[%d]: %s\n", i, gargv[i]);
  101. }
  102. }
  103. //执行命令
  104. bool ExecuteCommand()
  105. {
  106. pid_t id = fork();
  107. if (id < 0)
  108. return false;
  109. else if (id == 0)
  110. {
  111. // 子进程
  112. //执行命令
  113. execvpe(gargv[0], gargv,genv);
  114. //退出
  115. exit(1);
  116. }
  117. int status = 0;
  118. pid_t rid = waitpid(id, &status, 0);
  119. if (rid > 0)
  120. {
  121. if(WIFEXITED(status))
  122. {
  123. lastcode = WEXITSTATUS(status);
  124. }
  125. else//表示代码异常退出
  126. {
  127. lastcode = 100;
  128. }
  129. return true;
  130. }
  131. return false;
  132. }
  133. void AddEnv(const char *item)
  134. {
  135. int index = 0;
  136. while(genv[index])
  137. {
  138. index++;
  139. }
  140. genv[index] = (char*)malloc(strlen(item)+1);
  141. strncpy(genv[index],item,strlen(item)+1);
  142. index++;
  143. genv[index] = nullptr;
  144. }
  145. //在shell中
  146. //有些命令,必须由子进程执行
  147. //有些命令,不能由子进程执行,要由shell自己执行 -----内建命令 built command
  148. bool CheckAndExecBuiltCommand()
  149. {
  150. //不能让子进程进行,因为子进程退出就结束了,并不能影响下一个进程的工作路径
  151. if (strcmp(gargv[0], "cd") == 0)
  152. {
  153. if (gargc == 2)
  154. {
  155. chdir(gargv[1]);
  156. lastcode = 0;
  157. }
  158. else
  159. {
  160. lastcode = 1;
  161. }
  162. return true;
  163. }
  164. else if(strcmp(gargv[0],"export")==0)
  165. {
  166. if(gargc == 2)
  167. {
  168. AddEnv(gargv[1]);
  169. lastcode = 0;
  170. }
  171. else
  172. {
  173. lastcode = 2;
  174. }
  175. }
  176. else if(strcmp(gargv[0],"env") == 0)
  177. {
  178. for(int i = 0 ;genv[i];i++)
  179. {
  180. printf("%s\n",genv[i]);
  181. }
  182. lastcode = 0;
  183. return true;
  184. }
  185. else if(strcmp(gargv[0],"echo") == 0)
  186. {
  187. if(gargc == 2)
  188. {
  189. //echo $?
  190. //echo $PATH
  191. //echo hello
  192. if(gargv[1][0] == '$')
  193. {
  194. if(gargv[1][1] == '?')
  195. {
  196. printf("%d\n",lastcode);
  197. lastcode = 0;
  198. }
  199. }
  200. else
  201. {
  202. printf("%s\n",gargv[1]);
  203. lastcode = 0;
  204. }
  205. }
  206. else
  207. {
  208. lastcode = 3;
  209. }
  210. return true;
  211. }
  212. return false;
  213. }
  214. //作为一个shell,获取环境变量应该从系统环境变量获取
  215. //今天外面做不到就直接从父进程shell中获取环境变量
  216. void InitEnv()
  217. {
  218. extern char **environ;
  219. int index = 0;
  220. while (environ[index])
  221. {
  222. genv[index] = (char*)malloc(strlen(environ[index])+1);
  223. strncpy(genv[index],environ[index],strlen(environ[index]));
  224. index++;
  225. }
  226. genv[index] = nullptr;
  227. }
  228. int main()
  229. {
  230. //初始化环境变量表
  231. InitEnv();
  232. char command_buffer[basesize];
  233. while (true)
  234. {
  235. // 打印命令行提示符
  236. PrintCommandLine();
  237. // 获取用户命令
  238. if (!GetCommandLine(command_buffer))
  239. {
  240. continue;
  241. }
  242. // printf("%s\n", command_buffer);
  243. // 分析命令
  244. ParseCommandLine(command_buffer);
  245. // 判断是不是内建命令
  246. if (CheckAndExecBuiltCommand())
  247. {
  248. continue;
  249. }
  250. // debug();
  251. // 执行命令
  252. ExecuteCommand();
  253. }
  254. return 0;
  255. }
标签: linux 服务器 C++

本文转载自: https://blog.csdn.net/2302_80652761/article/details/144158984
版权归原作者 花糖纸木 所有, 如有侵权,请联系我们删除。

“【Linux】模拟实现命令行解释器shell”的评论:

还没有评论