基本shell功能实现(exec系列程序替换函数练习)

news2025/1/27 13:04:22

shell

  • 功能描述
  • 思路介绍
    • 1.实现常驻进程功能
    • 2.实现命令读取功能
    • 3. 实现命令解析功能
    • 4.实现子进程执行命令功能
    • 5.完善功能
  • 补充内容
    • 让父进程运行内置命令
    • 实现子进程能够获得父进程的环境变量功能(export命令)
    • shell实现重定向功能
  • 全部代码如下:

功能描述

实现一个类似于shell的命令行解释器。通过让子进程执行命令,父进程等待等待并解析命令,从而可以执行类似于“ls”,“ls -a -l -i”,'pwd"等linux指令。

思路介绍

1.实现常驻进程功能

在这里将要实现一个死循环,并且打印出提示信息。

while(1)
{
	//命令行解释器一定是一个常驻内存的进程,不退出
	//打印出提示信息[]
   23     printf("[xty@localhost myshell]#] ");
   24     fflush(stdout);
}

运行如图:
在这里插入图片描述

2.实现命令读取功能

使用fgets函数读取输入的内容,注意要注意把回车给删除(因为fgets会把回车也读取进来)

	 #define NUM 1024
	 //保存完整的命令字符串
	 char cmd_line[NUM];

     //2.获取用户键盘的输入[输入的是各种指令和选项:"ls -a -l -i"]
     memset(cmd_line, '\0', sizeof(cmd_line));
     if(fgets(cmd_line, sizeof cmd_line, stdin) == NULL)
     {
       continue;
     }
     cmd_line[strlen(cmd_line)-1] = '\0';
     //发现有"ls -a -l\n'\o'" 回车换行,我们应该删除这个
     //printf("echo:%s \n", cmd_line);

3. 实现命令解析功能

需要将我们输入的命令行字符串,变成shell能理解的语言。
将命令行选项使用strtok分开,把 "ls -a -l"变成 “ls”, “-a”, “-l”, “NULL”
为后面的execvp作准备。

#define SIZE 32
//保存打散之后的命令行字符串
char *g_argv[SIZE];

//3. 将命令行选项使用strtok分开, "ls -a -l"-> "ls", "-a", "-l", "NULL"
g_argv[0] = strtok(cmd_line, " ");
int index = 1;
while(g_argv[index++] = strtok(NULL, " "));


//检查一下g_argv对不对
for(index = 0; g_argv[index];index++)
{
  //虽然存入的是地址,但是%s,会将它看成字符串打印出来
  printf("g_argv[%d] = %s\n", index, g_argv[index]);
}

结果如下:
在这里插入图片描述

4.实现子进程执行命令功能

子进程执行命令,父进程等待子进程返回。

     //4.让子进程执行命令,执行完后给父进程返回值
     pid_t id = fork();
     if(id==0)
     {
       //子进程,执行命令 
       printf("子进程开始执行任务\n");                                                                     
       execvp(g_argv[0], g_argv);
       exit(1);
     }
    
     //father
     int status = 0;
     pid_t ret = waitpid(id, &status, 0);
     if(ret>0) printf("exit code:%d\n", WEXITSTATUS(status));

5.完善功能

让"ls"有颜色,并且让shell认识"ls"命令。
完整代码如下:

1 #include<stdio.h>
2 #include<unistd.h>
3 #include<stdlib.h>
4 #include<string.h>
5 #include<sys/wait.h>
6 #include<sys/types.h>
7 
8 #define NUM 1024
9 #define SIZE 32
10 //保存完整的命令字符串
11 char cmd_line[NUM];
12 //保存打散之后的命令行字符串
13 char *g_argv[SIZE];
14 
15 
16 //shell运行原理:通过让子进程执行命令,父进程等待&&解析命令
17 int main()
18 {                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          
19   //0.命令行解释器一定是一个常驻内存的进程,不退出
20   while(1)
21   {
22     //1.打印出提示信息[]
23     printf("[xty@localhost myshell]#] ");
24     fflush(stdout);
25 
26     //2.获取用户键盘的输入[输入的是各种指令和选项:"ls -a -l -i"]
27     memset(cmd_line, '\0', sizeof(cmd_line));
28     if(fgets(cmd_line, sizeof cmd_line, stdin) == NULL)
29     {
30       continue;
31     }
32     if(cmd_line[0] == '\n')
33     {
34       continue;
35     }
36     cmd_line[strlen(cmd_line)-1] = '\0';
37     //发现有"ls -a -l\n'\o'" 回车换行,我们应该删除这个
38     //printf("echo:%s \n", cmd_line);
39 
40     //3. 将命令行选项使用strtok分开, "ls -a -l"-> "ls", "-a", "-l", "NULL"
41     g_argv[0] = strtok(cmd_line, " ");
42     int index = 1;
43 
44     if(strcmp(g_argv[0], "ls")==0)
45     {
46       g_argv[index++]="--color=auto";
47     }
48     //还可以让编译器支"ll"
49     if(strcmp(g_argv[0],"ll")==0)
50     {
51       g_argv[0] = "ls";
52       g_argv[index++] = "-l";
53       g_argv[index++] = "--color=auto";
54     }
55 
56     //让ls命令有颜色
57     while(g_argv[index++] = strtok(NULL, " "));
58 
59     //检查一下g_argv对不对
60     for(index = 0; g_argv[index];index++)
61     {
62       //虽然存入的是地址,但是%s,会将它看成字符串打印出来
63       printf("g_argv[%d] = %s\n", index, g_argv[index]);
64     }
65 
66     //4.让子进程执行命令,执行完后给父进程返回值
67     pid_t id = fork();
68     if(id==0)
69     {
70       //子进程,执行命令 
71       printf("子进程开始执行任务\n");
72       execvp(g_argv[0], g_argv);
73       exit(1);
74     }
75 
76     //father
77     int status = 0;
78     pid_t ret = waitpid(id, &status, 0);
79     if(ret>0) printf("exit code:%d\n", WEXITSTATUS(status));
80   }
81   return 0;
82 }

这样我们一个简易的shell就完成了。

补充内容

shell执行的命令,通常有两种:

  1. 第三方提供的对应在磁盘中有具体二进制文件的可执行程序(由子进程执行)
  2. shell内部,自己实现的方法,由(父进程)来执行,有些命令会影响到shell本身,比如:“cd”,"export"命令等。

让父进程运行内置命令

当我们执行cd命令时,发现程序并没有改变目录。如下图:
原因是:cd命令被子进程执行了,子进程的当前目录被修改了,但是子进程立马就退出了。但是并没有影响到父进程的当前目录,所以第二次再次运行的时候,子进程还是在父进程的目录创建的,因此没有改变。
在这里插入图片描述

使用chdir函数改变父进程的工作目录。
代码如下:

   66     //让父进程执行cd 命令
   67     if(strcmp(g_argv[0], "cd")==0)
   68     {
   69       if(g_argv[1]!=NULL) chdir(g_argv[1]); // cd .., cd path
   70       continue;                                                                                                                                                   
   71     }    

结果如下图:
在这里插入图片描述

实现子进程能够获得父进程的环境变量功能(export命令)

因export和其他的命令行功能不一样,所以需要再判断一下该功能。并且需要创建新数组保存一下环境变量,因为cmd_line再第二次读取时会清空,会导致getenv时得不到被清空的环境变量(因为存的是指针地址,清空后变成了野指针),因此需要创建数组存储一下!

代码如下:


//myshell.c
 //写一个环境变量的bufer,用来存储临时的环境变量,防止cmd_line被清空
 char my_num[64];
 
 // export MYNUM=111222333
 if(strcmp(g_argv[0], "export")==0 && g_argv[1]!=NULL)
 {
   strcpy(my_num, g_argv[1]);
   int ret = putenv(my_num);
   if (ret == 0) printf("%s export sucess\n", my_num);  //查看一下导入的是什么
   continue;
 }
 

 //env_test.c
  1 #include<stdio.h>
  2 #include<stdlib.h>                                                                                                                                                  
  3 
  4 int main()
  5 {
  6   printf("我是测试环境变量是否成功导入进程\n");
  7   printf("MYNUM= %s \n",getenv("MYNUM"));
  8 
  9 }



在这里插入图片描述

shell实现重定向功能

先检查是否有重定向的功能,然后再执行命令。

//检查命令模块
    //定义标志位的含义                                                                                                                                                                                                                         
   22 #define INPUT_REDIR 1
   23 #define OUTPUT_REDIR 2
   24 #define APPEND_REDIR 3
   25 #define NONE_REDIR 0
   26 int redir_status = NONE_REDIR;
   27 
   28 char *CheckRedir(char *start)
   29 {
   30   assert(start);
   31   char *end = start + strlen(start) - 1;// ls -a -l
   32 
   33   //从后往前找
   34   while(end >= start)
   35   {
   36     if(*end == '>')
   37     {
   38       if(*(end - 1) == '>')
   39       {
   40         redir_status = APPEND_REDIR;
   41         *(end - 1) = '\0';
   42         end++;  //重定向后字符串的首地址
   43         break;
   44       }
   45       redir_status = OUTPUT_REDIR;
   46       *end = '\0';
   47       end++;   // ls -a>myfile.txt   ->ls -a\0myflie.txt
   48       break;
   49     }
   50     else if(*end == '<')
   51     {
   52       redir_status = INPUT_REDIR;
   53       *end = '\0';
   54       end++;
   55       break;
   56     }
   57     else{
   58       end--;
   59     }
   60   }
   61   if(end>=start)
   62   {
   63     return end; //要打开的文件名
   64   }
   65   else{
   66     return NULL;  //没有重定向功能
   67   }
   68 } 





		//main函数内部,子进程执行命令前,多一个重定向打开文件的逻辑:

  142     //4.让子进程执行命令,执行完后给父进程返回值
  143     pid_t id = fork();
  144     if(id==0)
  145     {
  146       //子进程,执行命令 
  147       printf("子进程开始执行任务\n");
  148       if(sep!=NULL)
  149       {
  150         int fd = -1;
  151         //有重定向
  152         switch(redir_status)
  153         {
  154           case INPUT_REDIR:
  155             fd = open(sep, O_RDONLY);
  156             dup2(fd, 0);
  157             break;
  158           case OUTPUT_REDIR:
  159             fd = open(sep, O_WRONLY | O_TRUNC | O_CREAT, 0666);
  160             dup2(fd, 1);
  161             break;
  162           case APPEND_REDIR:
  163             fd = open(sep, O_WRONLY | O_APPEND | O_CREAT, 0666);
  164             dup2(fd, 1);
  165             break;
  166           default:
  167             printf("erro????????\n");
  168             break;
  169         }
  170       }
  171       
  172       execvp(g_argv[0], g_argv);
  173       exit(1);
  174     }
  175    


全部代码如下:

    1 #include<stdio.h>
    2 #include<unistd.h>
    3 #include<stdlib.h>
    4 #include<string.h>
    5 #include<assert.h>
    6 #include<fcntl.h>
    7 #include<sys/wait.h>
    8 #include<sys/stat.h>
    9 #include<sys/types.h>
   10 
   11 #define NUM 1024
   12 #define SIZE 32
   13 //保存完整的命令字符串
   14 char cmd_line[NUM];
   15 //保存打散之后的命令行字符串
   16 char *g_argv[SIZE];
   17 
   18 //写一个环境变量的bufer,用来存储临时的环境变量,防止cmd_line被清空
   19 char my_num[64];
   20 
   21 //定义标志位的含义
   22 #define INPUT_REDIR 1
   23 #define OUTPUT_REDIR 2
   24 #define APPEND_REDIR 3
   25 #define NONE_REDIR 0
   26 int redir_status = NONE_REDIR;
   27 
   28 char *CheckRedir(char *start)
   29 {
   30   assert(start);
   31   char *end = start + strlen(start) - 1;// ls -a -l
   32   
   33   //从后往前找
   34   while(end >= start)
   35   {
   36     if(*end == '>')
   37     {
   38       if(*(end - 1) == '>')
   39       {
   40         redir_status = APPEND_REDIR; 
   41         *(end - 1) = '\0';
   42         end++;  //重定向后字符串的首地址
   43         break;
   44       }
   45       redir_status = OUTPUT_REDIR;
   46       *end = '\0';
   47       end++;   // ls -a>myfile.txt   ->ls -a\0myflie.txt
   48       break;
   49     }
   50     else if(*end == '<')
   51     {
   52       redir_status = INPUT_REDIR;
   53       *end = '\0';
   54       end++;
   55       break;
   56     }
   57     else{
   58       end--;
   59     }
   60   }
   61   if(end>=start)
   62   {
   63     return end; //要打开的文件名
   64   }
   65   else{
   66     return NULL;  //没有重定向功能
   67   }
   68 } 
   69 
   70 //shell运行原理:通过让子进程执行命令,父进程等待&&解析命令
   71 int main()
   72 {
   73   //0.命令行解释器一定是一个常驻内存的进程,不退出
   74   while(1)
   75   {
   76     //1.打印出提示信息[]
   77     printf("[xty@localhost myshell]#] ");
   78     fflush(stdout);
   79 
   80     //2.获取用户键盘的输入[输入的是各种指令和选项:"ls -a -l -i"]
   81     memset(cmd_line, '\0', sizeof(cmd_line));
   82     if(fgets(cmd_line, sizeof cmd_line, stdin) == NULL)
   83     {
   84       continue;
   85     }
   86     if(cmd_line[0] == '\n')
   87     {
   88       continue;
   89     }
   90     cmd_line[strlen(cmd_line)-1] = '\0';
   91     //发现有"ls -a -l\n'\o'" 回车换行,我们应该删除这个
   92     //printf("echo:%s \n", cmd_line);
   93     
   94 
   95     //在分析指令之前就检查有没有重定向
   96     char* sep = CheckRedir(cmd_line);
   97     
   98     //3. 将命令行选项使用strtok分开, "ls -a -l"-> "ls", "-a", "-l", "NULL"
   99     g_argv[0] = strtok(cmd_line, " ");
  100     int index = 1;
  101 
  102     if(strcmp(g_argv[0], "ls")==0)
  103     {
W>104       g_argv[index++]="--color=auto";
  105     }
  106     //还可以让编译器支"ll"
  107     if(strcmp(g_argv[0],"ll")==0)
  108     {
W>109       g_argv[0] = "ls";
W>110       g_argv[index++] = "-l";
W>111       g_argv[index++] = "--color=auto";
  112     }
  113 
  114     //让ls命令有颜色
W>115     while(g_argv[index++] = strtok(NULL, " "));
  116     
  117     检查一下g_argv对不对
  118     //for(index = 0; g_argv[index];index++)
  119     //{
  120     //  //虽然存入的是地址,但是%s,会将它看成字符串打印出来
  121     //  printf("g_argv[%d] = %s\n", index, g_argv[index]);
  122     //}
  123 
  124 
  125     // export MYNUM=111222333
  126     if(strcmp(g_argv[0], "export")==0 && g_argv[1]!=NULL)
  127     {
  128       strcpy(my_num, g_argv[1]);
  129       int ret = putenv(my_num);
  130       if (ret == 0) printf("%s export sucess\n", my_num);  //查看一下导入的是什么
  131       continue;
  132     }
  133 
  134 
  135 
  136     //让父进程执行cd 命令
  137     if(strcmp(g_argv[0], "cd")==0)
  138     {
  139       if(g_argv[1]!=NULL) chdir(g_argv[1]); // cd .., cd path
  140       continue;
  141     }
  142     //4.让子进程执行命令,执行完后给父进程返回值
  143     pid_t id = fork();
  144     if(id==0)
  145     {
  146       //子进程,执行命令 
  147       printf("子进程开始执行任务\n");
  148       if(sep!=NULL)
  149       {
  150         int fd = -1;
  151         //有重定向
  152         switch(redir_status)
  153         {
  154           case INPUT_REDIR:
  155             fd = open(sep, O_RDONLY);
  156             dup2(fd, 0);
  157             break;
  158           case OUTPUT_REDIR:
  159             fd = open(sep, O_WRONLY | O_TRUNC | O_CREAT, 0666);
  160             dup2(fd, 1);
  161             break;
  162           case APPEND_REDIR:
  163             fd = open(sep, O_WRONLY | O_APPEND | O_CREAT, 0666);
  164             dup2(fd, 1);
  165             break;
  166           default:
  167             printf("erro????????\n");
  168             break;
  169         }
  170       }
  171 
  172       execvp(g_argv[0], g_argv);
  173       exit(1);
  174     }
  175 
  176     //father
  177     int status = 0;
  178     pid_t ret = waitpid(id, &status, 0);
  179     if(ret>0) printf("exit code:%d\n", WEXITSTATUS(status));
  180   }
  181   return 0;
  182 }                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                  

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

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

相关文章

TCP协议:可靠传输的基石

目录 1. 数据完整性的保证 2. 数据的有序传输 3. 确认应答机制 4. 流量控制 5. 拥塞控制 6. 重传机制 结论 引言 TCP&#xff08;Transmission Control Protocol&#xff09;是计算机网络中的一个重要协议&#xff0c;它以其可靠性而闻名。TCP是一种面向连接的协议&…

Hive执行计划

Hive提供了explain命令来展示一个查询的执行计划&#xff0c;这个执行计划对于我们了解底层原理&#xff0c;Hive 调优&#xff0c;排查数据倾斜等很有帮助。 使用语法如下&#xff1a; explain query;在 hive cli 中输入以下命令(hive 2.3.7)&#xff1a; explain select s…

Hadoop 集群环境搭建

目录 第一部分&#xff1a;系统安装... 3 1&#xff1a;图形化安装... 3 2&#xff1a;选择中文... 3 3&#xff1a;安装选项... 3 4&#xff1a;软件选项... 4 5&#xff1a;安装位置... 4 6&#xff1a;网络配置... 6 7&#xff1a;开始安装... 7 8&#xff1a;创建用户... 7…

oracle怎样才算开启了内存大页?

oracle怎样才算开启了内存大页&#xff1f; 关键核查下面三点&#xff1a; 1./etc/sysctl.conf vm.nr_hugepages16384这是给了32G&#xff0c;计划sga给30G&#xff0c;一般需多分配2-4G sysctl -p生效 看cat /proc/meminfo|grep Huge啥结果&#xff1f; 这种明显是配了…

海康威视对讲广播系统 RCE漏洞复现(CVE-2023-6895)

0x01 产品简介 Hikvision Intercom Broadcasting System是中国海康威视(Hikvision)公司的一个对讲广播系统。 0x02 漏洞概述 Hikvision Intercom Broadcasting System 3.0.3_20201113_RELEASE(HIK)版本存在操作系统命令注入漏洞,该漏洞源于文件/php/ping.php的参数jsonda…

WooCommerce Step Filter商城网站步骤过滤器

点击阅读WooCommerce Step Filter商城网站步骤过滤器详情 WooCommerce Step Filter商城网站步骤过滤器是强大而灵活的插件&#xff0c;用于为您的产品创建不同类型的过滤器。它可以通过回答问题和给出每个步骤的描述来帮助您的客户始终如一地选择产品。或者只是将其用作小部件…

Ubuntu20.04.2-mate上Lazarus安装与测试

简言 Lazarus采用RAD方式界面开发&#xff0c;一套代码可交差编译出windows、ios、android、solaris、BSD等 各平台运行的程序&#xff0c;在unbuntu的repo中有2.2.0版本可用&#xff0c;在sourceforge上有2.2.6版本和3.0.0的Rolling版可下载安装&#xff0c;但感觉上2.2.0和2…

基于华为atlas的烟火检测实战

1、下载官方yolov5的v6.1版本 git clone https://github.com/ultralytics/yolov5.git git checkout v6.1 2、烟火数据集准备&#xff1a; tree -d Images/train/目录下图片 Labels/train/目录下标签 3、数据格式转化&#xff1a; 数据集采用labelimg标注&#xff0c;xml文件…

利用prometheus+grafana进行Linux主机监控

文章目录 一.架构说明与资源准备二.部署prometheus1.上传软件包2.解压软件包并移动到指定位置3.修改配置文件4.编写启动脚本5.启动prometheus服务 三.部署node-exporter1.上传和解压软件包2.设置systemctl启动3.启动服务 四.部署grafana1.安装和启动grafana2.设置prometheus数据…

python画图【01】

前提&#xff1a;使用anaconda环境&#xff0c;且安装好&#xff0c;使用的是jupyter pandas 和 matplotlib 安装教程可以参考&#xff1a;miniconda安装与使用 import pandas as pd读取xlsx表格数据 data pd.read_excel("data1.xlsx",sheet_nameSheet1) #data p…

Ubuntu中文本编辑器和编译器

你好&#xff0c;这里是争做图书馆扫地僧的小白。 个人主页&#xff1a;争做图书馆扫地僧的小白_-CSDN博客 目标&#xff1a;希望通过学习技术&#xff0c;期待着改变世界。 目录 前言 一、vim编辑器 &#xff08;一&#xff09;打开vim编辑器 &#xff08;二&#xff09;v…

Opencv 入门三(视频滑动条窗口)

视频滑动条窗口源码如下&#xff1a; #include "opencv2\highgui\highgui.hpp" #include "opencv2/imgproc/imgproc.hpp" #include <iostream> #include <fstream> using namespace std; int g_slider_position 0; // 滑动条的位置 …

转行小白历险记-2023.12.19 如何解决跨域问题

讨厌突如其来的破坏计划的感觉&#xff0c;真的好烦 一、跨域以及如何解决跨域问题(反向代理) 跨域 同源&#xff1a;协议、端口、域名相同 如何解决跨域问题 jsonp:<script> 不受同源策略限制跨源域资源共享 CORS:允许 Web 应用服务器进行跨源访问控制使不同的源变成同…

CentOS6.10 卸载MySQL8.0.34升级至8.0.35

准备要更新的MySQL安装包,下载地址:MySQL :: Download MySQL Community Server 查看当前MySQL版本 备份数据库 mysqldump -uroot -p -B > /opt/backup/20231220_mysql.sql 检查备份文件 查看所有服务项: service --status-all 可以看到我们注册的MySQL服务是mysqld 停止…

【第七在线】供应链协作与商品计划:建立强大的合作关系

供应链协作在现代服装企业的商品计划中扮演着至关重要的角色。建立强大的合作关系能够提高生产效率、降低成本、减少库存和提供更好的客户服务。本文将深入探讨供应链协作的重要性&#xff0c;以及如何在服装企业中建立和维护这种关键关系。 1. 供应链协作的背景 供应链协作是…

使用Python爬取GooglePlay并从复杂的自定义数据结构中实现解析

文章目录 【作者主页】&#xff1a;吴秋霖 【作者介绍】&#xff1a;Python领域优质创作者、阿里云博客专家、华为云享专家。长期致力于Python与爬虫领域研究与开发工作&#xff01; 【作者推荐】&#xff1a;对JS逆向感兴趣的朋友可以关注《爬虫JS逆向实战》&#xff0c;对分布…

高速视频采集卡设计方案:620-基于PCIe的高速视频采集卡

一、产品概述 基于PCIe的高速视频采集卡&#xff0c;通过PCIe3.0X8传输到存储计算服务器&#xff0c;实现信号的分析、存储。 北京太速科技 产品固化FPGA逻辑&#xff0c;适配视频连续采集&#xff0c;缓存容量2GB&#xff0c;开源的PCIe QT客户端软件&#xff0c…

Kafka--Kafka日志索引详解以及生产常见问题分析与总结

一、Kafka的Log日志梳理 ​ 这一部分数据主要包含当前Broker节点的消息数据(在Kafka中称为Log日志)。这是一部分无状态的数据&#xff0c;也就是说每个Kafka的Broker节点都是以相同的逻辑运行。这种无状态的服务设计让Kafka集群能够比较容易的进行水平扩展。比如你需要用一个新…

数据库原理及应用·数据库系统结构

2.1 数据模型的概念 2.1.1 什么是数据模型 数据模型&#xff08;Data Model&#xff09; 是对现实世界数据特征的模拟和抽象&#xff0c;用来描述数据是如何组织、存储和操作的。 数据模型应满足如下三个条件&#xff1a; 能比较真实地模拟现实世界 容易为人所理解 便于在计…

设计模式(三)-结构型模式(3)-装饰模式

一、为何需要装饰模式&#xff08;Decorator&#xff09;? 在软件设计中&#xff0c;某个对象会组合很多不同的功能&#xff0c;如果把所有功能都写在这个对象所在的类里&#xff0c;该类会包含很多复杂的代码逻辑&#xff0c;导致代码不美观且难以维护。于是就有了再定义一些…