C++面向对象程序设计 - 运算符重载

        函数重载就是对一个已有的函数赋予新的含义,使之实现新的功能。因此一个函数名就可以用来代表不同功能的函数,也就是一名多用。运算符也可以重载,即运算符重载(operator overloading)。

一、运算符重载的方法

        运算符重载的方法是定义一个重载运算符的函数,在需要执行被重载的运算符时,系统就自动调用该函数,以实现相应的运算。运算符重载实质上是函数的重载。

        重载运算符的函数一般格式如下:

函数类型  operator 运算符名称(形参表列){

        对运算符的重载处理

}

        这里通过 + 运算符进行两个复数相加,示例代码如下:

#include <iostream>
using namespace std;
class Complex{public:Complex(){real = 0; imag = 0;}Complex(double r, double i){real = r; imag = i;}Complex operator +(Complex &c2);	// 声明重载运算符 + 的函数void display();private:double real;double imag;
};
// 定义重载运算符 + 的函数
Complex Complex::operator +(Complex &c2){Complex c;c.real = real + c2.real;		// 两上复数的实部相加c.imag = imag + c2.imag;		// 两个复数的虚部相加return c;
}
void Complex::display(){cout <<"("<<real<<","<<imag<<"i)"<<endl;
}
int main(){Complex c1(3, 4), c2(5, -10), c3;c3 = c1 + c2;			// 运算符 + 用于复数运算cout <<"c1=";c1.display();cout <<"c2=";c2.display();cout <<"c1+c2=";c3.display();return 0;
}

运行结果如下:

        对于上述示例的运算符重载函数 operator + 还可以改写得更简练一些,代码如下:

#include <iostream>
using namespace std;
class Complex{public:Complex(){real = 0; imag = 0;}Complex(double r, double i){real = r; imag = i;}Complex operator +(Complex &c2);	// 声明重载运算符 + 的函数void display();private:double real;double imag;
};
// 定义重载运算符 + 的函数
Complex Complex::operator +(Complex &c2){return Complex(real + c2.real, imag + c2.imag);
}
void Complex::display(){cout <<"("<<real<<","<<imag<<"i)"<<endl;
}
int main(){Complex c1(3, 4), c2(5, -10), c3;c3 = c1 + c2;			// 运算符 + 用于复数运算cout <<"c1=";c1.display();cout <<"c2=";c2.display();cout <<"c1+c2=";c3.display();return 0;
}

        return语句中的Complex(real + c2.real, imag + c2.imag) 是建立一个临时对象,它是一个无名对象,在建立临时对象过程中调用构造函数,return语句将此临时对象作为函数返回值。

二、重载运算符的规则

        1)C++不允许用户自己定义新的运算符,只能对已有的C++运算符进行重载。

        2)C++中绝大部分的运算符允许重载。如下表

双目算术运算符+(加),-(减),*(乘),/(除),%(取模)
关系运算符==(等于),!=(不等于),<(小于),>(大于),<=(小于等于),>=(大于等于)
逻辑运算符||(逻辑或),&&(逻辑与),!(逻辑非)
单目运算符+(正),-(负),*(指针),&(取地址)
自增自减运算符++(自增),--(自减)
位运算符!(按位或),&(按位与),~(按位取反),^(按位异或),<<(左移),>>(右移)
赋值运算符=,+=,-=,*=,/=,%=,&=,!=,^=,<<=,>>=
空间申请与释放new,delete,new[],delete[]
其他运算符()(函数调用),->(成员访问),->*(成员指针访问),,(逗号),[](下标)

        不能重载的运算符:

.成员访问运算符
.*成员指针访问运算符
::域运算符
sizeof长度运算符
?:条件运算符

        3)重载不能改变运算符运算对象(即操作数)的个数;例如 + 运算符有两个操作数(a + b),不能重载它使用得它只有一个操作数或三个操作数。

        4)重载不能改变运算符的优先级别。重载不能改变运算符的结合性;例如 * 和 / 优先于 + 和 - ,不论怎样进行重载,各大运算符的优先级别不会改变。

        5)重载不能改变运算符的结合性。结合性决定了当多个相同优先级的运算符出现在同一个表达式中时,应该如何分组。例如 + 和 * 都是左结合的,这意味着 a + b + c会被解释为 (a + b) + c,而不是 a + (b + c)。重载运算符不能改变这种结合性。

        6)重载运算符的函数不能有默认的参数,这是因为默认参数会改变运算符的语义,导致代码的混淆和不可预测的行为。

        7)重载的运算符必须和用户定义的自定义类型的对象一起使用,其参数至少应有一个类对象(或类对象的引用)。重载运算符的目的是为了增强自定义类型的操作能力,而不是改变内置类型的行为。因此,重载的运算符必须至少涉及一个自定义类型的对象。

        8)用于类对象的运算符一般必须重载,但有两个例外,运算符 = 和 & 不必用户重载。

        9)应当使重载运算符的功能类似于该运算符作用于标准类型数据时所实现的功能。

        10)运算符重载函数可以是类的成员函数,也可以是类的友元函数,还可以是既非类的成员函数也不是友元函数的普通函数。

        总之,这些规则确保了运算符重载的一致性和可预测性,同时也避免了可能的混淆和错误。

三、运算符重载函数作为类成员函数和友元函数

        运算符重载函数除了可以作为类的成员函数,还可以是非成员函数。此时将运算符“+”重载为适用于复数相加,重载函数不作为成员函数,而放在类外作为Complex类的友函数。代码如下:

#include <iostream>
using namespace std;
class Complex{public:Complex(){real = 0; imag = 0;}Complex(double r, double i){real = r; imag = i;}// 声明重载运算符 + 的函数friend Complex operator +(Complex &c1, Complex &c2);	void display();private:double real;double imag;
};
// 定义重载运算符 + 的函数
Complex operator +(Complex &c1, Complex &c2){return Complex(c1.real + c2.real, c1.imag + c2.imag);
}
void Complex::display(){cout <<"("<<real<<","<<imag<<"i)"<<endl;
}
int main(){Complex c1(3, 4), c2(5, -10), c3;c3 = c1 + c2;			// 运算符 + 用于复数运算cout <<"c1=";c1.display();cout <<"c2=";c2.display();cout <<"c1+c2=";c3.display();return 0;
}

        运行结果如下图:

四、重载双目运算符

        双目运算符(或称二元运算符)是C++中最常用的运算符,双目运算符有两个操作数,在运算符的左右两侧,如 3 + 5, a = b,i < 10等。

        这里先给一个示例,重载运算符 > ,< ,=等,代码如下:

#include <iostream>
#include <string>
using namespace std;class String{public:String(){p = NULL;}String(const char *str){p = str;}friend bool operator > (String &str1, String &str2);friend bool operator < (String &str1, String &str2);friend bool operator == (String &str1, String &str2);private:const char *p;
};
// 重载运算符 >
bool operator > (String &str1, String &str2){return strcmp(str1.p, str2.p)>0 ? true : false;
}
// 重载运算符 <
bool operator < (String &str1, String &str2){return strcmp(str1.p, str2.p)<0 ? true : false;
}
// 重载运算符 ==
bool operator == (String &str1, String &str2){return strcmp(str1.p, str2.p)==0 ? true : false;
}int main(){String s1("Hello"), s2("World"), s3("Computer");cout <<(s1 < s2) <<endl;cout <<(s1 > s3) <<endl;cout <<(s2 == s3) <<endl;return 0;
}

        运算结果如下:

        需要注意的是此示例中是将字符串转换为一个char*类型, 在C++中,字符串常量是const char*类型(例如“Hello”,"World"),而将其直接赋值给char* 类型的变量会导致编译器报错【[Warning] ISO C++ forbids converting a string constant to 'char*' [-Wwrite-strings]】- C++禁止针字符串常量转换为'char*'。将只读的字符串常量赋值给char*类型修改为可变的内容,这是不被允许的,所以在定义String类中的私有数据成中和构造函数中形参声明,一定要在前面另上const。

        下面将以上代码进行优化,让显示的结果更为直观些,代码如下:

#include <iostream>
#include <string>
using namespace std;class String{public:String(){p = NULL;}String(const char *str){p = str;}friend bool operator > (String &str1, String &str2);friend bool operator < (String &str1, String &str2);friend bool operator == (String &str1, String &str2);friend void compare(String &str1, String &str2);void display(){cout <<p;}private:const char *p;
};
// 重载运算符 >
bool operator > (String &str1, String &str2){return strcmp(str1.p, str2.p)>0 ? true : false;
}
// 重载运算符 <
bool operator < (String &str1, String &str2){return strcmp(str1.p, str2.p)<0 ? true : false;
}
// 重载运算符 ==
bool operator == (String &str1, String &str2){return strcmp(str1.p, str2.p)==0 ? true : false;
}
// 更直观显示结果
void compare(String &str1, String &str2){if(operator > (str1, str2) == 1){cout <<str1.p <<'>' <<str2.p;} else{if(operator < (str1, str2) == 1){cout <<str1.p <<'<' <<str2.p;} else if(operator == (str1, str2) == 1){cout <<str1.p <<'=' <<str2.p;}}cout <<endl;
}int main(){String s1("Hello"), s2("World"), s3("Computer");compare(s1, s2);compare(s1, s3);compare(s2, s3);return 0;
}

        运行结果如下:

        此时需要区分下strcmp 和 operator >,在c++中strcp是C标准库的函数,用于比较两个字符中(即char*指针向的字符串);而operator >是一个运算符重载,可以在自定义的类中定义,用于比较对象。

        strcmp函数返回一个整数,表示两上字符串的字典关系:

  • 如果返回值小于0,表示值1指向的字符串在字典上小于值2指向的字符串。
  • 如果返回值等于0,表示两个字符串相等。
  • 如果返回值大于0,表示值1指向的字符串在字典上大于值2指向的字符串。

        示例中operator > 或 operator < 或 operator == 是String类中声明的3个重载函数为友元函数,并在类外分别定义了3个运算符重载函数。C++编译系统将程序中的表达式 s1 > s2 解释为 operator > (s1, s2),所以上例代码中写法也就不难理解了。将compare函数也可以改为以下形式,代码如下:

void compare(String &str1, String &str2){if((str1 > str2) == 1){cout <<str1.p <<'>' <<str2.p;} else{if((str1 < str2) == 1){cout <<str1.p <<'<' <<str2.p;} else if((str1 == str2) == 1){cout <<str1.p <<'=' <<str2.p;}}cout <<endl;
}

五、重载单目运算符

        单目运算符只有一上操作数,如 !a,-b,&c,*p,还有常用的 ++i 和--i等。重载单目运算符的方法与重载双目运算符的方法类似的。但由于单目运算符只有一个操作数,因此运算符重载函数只有一个参数,如果运算符重载函数作为成员函数,则还可以省略此参数。

        这里通过一个计算器Timer类来演示一下单目运算符的重载,代码如下:

#include <iostream>
using namespace std;
class Timer{public:Timer(){hour = 0;minute = 0;second = 0;}Timer(int h, int m, int s): hour(h), minute(m), second(s){}Timer operator ++ ();		// 声明运算符重载// 定义输出时间函数void display(){cout <<hour <<":" <<minute <<":" <<second <<endl;}private:int hour;int minute;int second;
};
Timer Timer::operator ++(){// 分钟 满60秒进1分钟if(++second>=60) {second -= 60;++minute;}// 小时 满60分钟进1小时if(minute>=60){minute -= 60;++hour;}// 返回当前对象return *this;
}
int main(){Timer t1(1, 20, 0);for(int i = 0; i < 61; i++){++t1;t1.display();}return 0;
}

        运行一共输出61行,结果如下:

1:20:1
1:20:2
1:20:3
1:20:4
1:20:5
1:20:6

......
1:20:56
1:20:57
1:20:58
1:20:59
1:21:0
1:21:1

       在C++中,如果在自增(自减)运算符重载函数中,增加一个int型形参,就是后后置自增(自减)运算符函数。示例如下:

#include <iostream>
using namespace std;
class Timer{public:Timer(){hour = 0;minute = 0;second = 0;}Timer(int h, int m, int s): hour(h), minute(m), second(s){}Timer operator ++ ();		// 声明前置自增运算符重载函数Timer operator ++ (int);	//声明后置自增运算符重载函数// 定义输出时间函数void display(){cout <<hour <<":" <<minute <<":" <<second <<endl;}private:int hour;int minute;int second;
};
// 定义前置自增运算符重载函数
Timer Timer::operator ++(){// 分钟 满60秒进1分钟if(++second>=60) {second -= 60;++minute;}// 小时 满60分钟进1小时if(minute>=60){minute -= 60;++hour;}// 返回当前对象return *this;
}
// 定义后置自增运算符重载函数
Timer Timer::operator ++(int){// 分钟 满60秒进1分钟if(++second>=60) {second -= 60;++minute;}// 小时 满60分钟进1小时if(minute>=60){minute -= 60;++hour;}// 返回当前对象return *this;
}
int main(){Timer t1(1, 20, 0), t2(3, 10, 20);for(int i = 0; i < 61; i++){++t1;t1.display();}cout <<endl;for(int i = 0; i < 61; i++){t2++;t2.display();}return 0;
}

        运行结果如下:

// 前置结果

1:20:1
1:20:2
1:20:3
1:20:4
1:20:5
1:20:6
.......
1:20:56
1:20:57
1:20:58
1:20:59
1:21:0
1:21:1

// 后置结果

3:10:21
3:10:22
3:10:23
3:10:24
3:10:25
3:10:26
.......
3:11:17
3:11:18
3:11:19
3:11:20
3:11:21

        注意的时,如果Timer类中未声明和定义后置自增运算符重载,则系统编译时会报错【[Error] no 'operator++(int)' declared for postfix '++' [-fpermissive]】- 没有为后缀‘++’声明'operator++(int)'。

        "++"或"--"前置和后置的区别:前置是先自加,返回是修改后的对象本身;后置是返回自加前的对象,然后对象再自加。

六、重载流插入运算符和提取运算符

        C++的插入运算符“<<”和流提取运算符“>>”是C++在类库中提供的,所有C++编译系统都在类库中提供输入流类istream和输出流类otream。cin和cout分别是istream和ostream类的对象。在类库提供的头文件中已经对"<<"和">>"进行了重载,使之作为流插入运算符和流提取运算符,能用来输出和输入C++标准类型的数据。

        对"<<"和">>"重载的函数形式如下:

istream & operator >> (istream &, 自定义类 &);

ostream & operator << (ostream &, 自定义类 &);

        第一个参数必须是流类型引用,第二个参数是要进行操作的类,所以重载"<<"和">>"的函数作为友函数或普通的函数,而不能将它们定义为成员函数。

        这是因为作为成员函数,这些运算符只能访问一个对象(即该成员函数所属的类的实例),然而在流操作中,我们需要同时访问流对象和用户定义的对象。如果试图将 << 和 >> 作为成员函数重载,那么它只能访问到自定义的对象,而无法访问到流对象,这显示是无法满足需求的。

        通过将 << 和 >> 重载为非成员函数,可以让它们同时接受一个流对象和一个自定义的对象作为参数,这样就可以在函数中同时访问和操作这两个对象。此外,为了允许这些非成员函数访问类的私有或保护成员,通常需要将这些函数声明为类的友元函数。

        这里将在“一”、“二”中的基础上,用重载的">>"输入数值,"<<"输出复数,代码如下:

#include <iostream>
using namespace std;
class Complex{public:Complex(){real = 0; imag = 0;}Complex(double r, double i){real = r; imag = i;}Complex operator +(Complex &c2);	// 声明重载运算符 + 的函数friend ostream& operator <<(ostream&, Complex&);friend istream& operator >>(istream&, Complex&);private:double real;double imag;
};
// 定义重载运算符 + 的函数
Complex Complex::operator +(Complex &c2){return Complex(real + c2.real, imag + c2.imag);
}
// 定义运算符">>"重载函数
istream& operator >>(istream& input, Complex& c){cout <<"Please enter the value:" <<endl;input >>c.real >>c.imag;return input;
}
// 定义运算符"<<"重载函数
ostream& operator <<(ostream& output, Complex& c){cout <<"("<<c.real<<","<<c.imag<<"i)"<<endl; return output;
}
int main(){Complex c1, c2, c3;cin >>c1 >>c2;c3 = c1 + c2;			// 运算符 + 用于复数运算cout <<endl;cout <<"c1=" <<c1;cout <<"c2=" <<c2;cout <<"c1+c2=" <<c3;return 0;
}

        运行后则会按运算符">>"重载后格式输入,以及运算符"<<"重载后格式输出,结果如下图:

        在C++中,运算符重载是很重要的、很有实用意义的,它使类的设计更加丰富多彩,扩大了类的功能和使用范围,使程序易于理解,易于对对象进行操作,它体现了为用户着想、方便用户使用的思路。

        也能体现到运算符重载中使用引用(reference)的重要性,利用引用作为函数形参可以不生成临时变量,减少了时间和空间的开销,通过传址的方式使形参成为实参的别名,调用过程中不是用传递值的方式 进行虚实结合。

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

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

相关文章

indexDB 大图缓存

背景 最近在项目中遇到了一个问题&#xff1a;由于大屏背景图加载速度过慢&#xff0c;导致页面黑屏时间过长&#xff0c;影响了用户的体验。从下图可以看出加载耗时将近一分钟 IndexDB 主要的想法就是利用indexDB去做缓存&#xff0c;优化加载速度&#xff1b;在这之前&am…

在选择试验台底座时,应注意哪些问题——河北北重

在选择试验台底座时&#xff0c;应注意以下几个方面&#xff1a; 底座尺寸和承载能力&#xff1a;底座的尺寸和承载能力应与试验台的尺寸和所需承载的设备重量相匹配&#xff0c;确保底座能够稳定承载试验台和设备。 材料和质量&#xff1a;底座的材料应具有足够的强度和耐久性…

软件测试之【合理的利用GPT来辅助软件测试一】

读者大大们好呀&#xff01;&#xff01;!☀️☀️☀️ &#x1f525; 欢迎来到我的博客 &#x1f440;期待大大的关注哦❗️❗️❗️ &#x1f680;欢迎收看我的主页文章➡️寻至善的主页 文章目录 前言GPT的原理及技巧GPT辅助接口自动化测试 前言 在编程基础栏目中&#xff…

Kafak详解(1)

简介 消息队列 为什么要有消息队列 图-1 消息队列的使用 消息队列 1)消息Message&#xff1a;网络中的两台计算机或者两个通讯设备之间传递的数据。例如说&#xff1a;文本、音乐、视频等内容。 2)队列Queue&#xff1a;一种特殊的线性表(数据元素首尾相接)&#xff0c;特…

2024平替电容笔买哪个品牌好?iPad电容笔全能榜单热门款TOP5分享!

2024年&#xff0c;随着科技的不断发展和消费者对生活品质的追求&#xff0c;电容笔作为一种创新的无纸化工具&#xff0c;逐渐走进人们的生活和工作中。然而&#xff0c;在电容笔市场的繁荣背后&#xff0c;也隐藏着品质良莠不齐的现象。众多品牌为了追求利润&#xff0c;推出…

JS----随机数字,字符,数组

随机数字 function random(min 0, max 100) {if (min > 0 && max > 0 && max > min) {const gab max - min 1return Math.floor(Math.random() * gab min)}return 0 }输入格式 随机字符 function randomStr (len 32) {var s for (; s.lengt…

电力调度自动化系统由什么构成?

电力调度自动化系统由什么构成&#xff1f; 电力调度自动化系统通过数据采集与传输、数据处理与存储、监视与控制、优化与决策、通信网络和系统应用软件等构成&#xff0c;实现对电力系统的监控、控制和优化。 电力调度自动化系统是一种集成了计算机技术、通信技术、自动化技术…

SpringBoot下载Excel模板功能

目录 一、前端只需要填写一个a标签调用一下后端接口即可 二、后端 2.1 准备一个excel模板 &#xff0c;将其复制到resource目录下的templates文件夹下 2.2 接着复制下列代码即可 三、运行效果 一、前端只需要填写一个a标签调用一下后端接口即可 1.1 先代理一下防止跨域 e…

40-50W 1.5KVDC 隔离 宽电压输入 DC/DC 电源模块 ——TP40(50)DC 系列

TP40(50)DC系列电源模块额定输出功率为40-50W、应用于2:1、4&#xff1a;1电压输入范围 9V-18V、18V-36V、36V-75V、9V-36V、18V-75V的输入电压环境&#xff0c;输出电压精度可达1%&#xff0c;可广泛应用于通信、铁路、自动化以及仪器仪表等行业。

【系统分析师】软件工程

文章目录 1、信息系统生命周期2、软件开发模型2.1 原型及其演化2.2 增量模型和螺旋模型2.3 V模型、喷泉模型、快速应用开发2.4 构件组装模型2.5 统一过程-UP2.6 敏捷方法 3、逆向工程4、净室软件工程 【写在前面】 记录了一系列【系统分析师】文章&#xff0c;点击下面的链接&a…

【鸿蒙开发】后台任务

1. 功能介绍 设备返回主界面、锁屏、应用切换等操作会使应用退至后台。 2. 后台任务类型 OpenHarmony标准系统支持规范内受约束的后台任务&#xff0c;包括短时任务、长时任务、延迟任务、代理提醒和能效资源。 开发者可以根据如下功能介绍&#xff0c;选择合适的后台任务以…

软件测试架构体系之软件测试基本流程图

前言&#xff1a; 采用通用的测试流程&#xff0c;能高效、高质量的完成软件测试工作&#xff0c;有助于减少沟通成本&#xff0c;对各阶段产出有明确认知等等。最终目标&#xff1a;实现软件测试规范化、标准化。以下为非通用标准&#xff0c;仅供大家参考。 一、软件测试流…

DSView Windows平台编译

在Windows平台编译开源逻辑分析仪软件DSView&#xff0c;因官方没有公布DSView Windows平台源码&#xff0c;主要解决Windows平台以下问题&#xff1a; libusb_get_pollfds不支持Windows平台&#xff0c;导致无法采集数据插入设备后&#xff0c;无法自动识别设备&#xff0c;U…

vue3中web前端JS动画案例(一)

上述案例主要使用定时器&#xff0c;和绝对定位产生动画 <script setup> import { ref, onMounted, watch } from vue // ----------------------- 01 js 动画介绍--------------------- // 1、匀速运动 // 2、缓动运动&#xff08;常见&#xff09; // 3、透明度运动 //…

【深度学习实战(18)】torch模型转onnx模型,使用netron根据查看onnx结构

一、ONNX介绍 简单来说&#xff0c;可以把ONNX当做一个中间格式。绝大多数的机器学习/深度学习框架都可以将自身的模型转换成ONNX&#xff0c;同样也能把ONNX转换成自身框架的格式&#xff0c;如下图所示。 二、转ONNX模型代码 import torch import torchvision.models as …

利用selenium发挥vip残存的价值

历史版本谷歌浏览器驱动下载地址 https://chromedriver.storage.googleapis.com/index.html 找到与你电脑当前谷歌浏览器版本一致的驱动然后下载下来(大版本一致即可)。我本地版本是 99.0.04844.51 我这里把 chromedriver 放到 /usr/local/bin 下面了。 启动测试窗口 这里需要…

[tkinter实现]汉字笔顺小软件

软件简介 本软件旨在帮助小学生通过互动式学习掌握汉字的基本笔画和笔顺。软件采用Tkinter库构建&#xff0c;提供了一个用户友好的图形界面&#xff0c;适合小学生使用。 主要功能&#xff1a; 汉字展示&#xff1a;软件能够展示单个汉字&#xff0c;并以动画形式演示其标准…

网络基础(day2)

一、进制转换 在计算机底层通信&#xff0c;以及数据处理都是采用二进制&#xff0c;也就是1和0传递信息&#xff0c;因此进制转换是非常重要的知识点。 十进制&#xff08;DEC&#xff09;【逢十进一 】案例&#xff1a; 按权展开 权&#xff1a;10的N次幂123123二进制&am…

羊大师分析,夏季羊奶的适合人群有哪些?

羊大师分析&#xff0c;夏季羊奶的适合人群有哪些&#xff1f; 夏季羊奶的适合人群相当广泛&#xff0c;主要包括以下几类人群&#xff1a; 生长发育中的孩子&#xff1a;羊奶富含营养&#xff0c;特别是蛋白质和矿物质&#xff0c;对孩子的生长发育有积极的促进作用。 中老年…

【HarmonyOS】Stage 模型 - 基本概念

一、项目结构 如图1所示&#xff1a; 图1 从项目结构来看&#xff0c;这个应用的内部包含了一个子模块叫 entry&#xff0c;模块是应用的基本功能单元&#xff0c;它里面包含源代码、资源、配置文件等。 像这样的模块在应用内部可以创建很多。但模块整体来讲就分成两大类&am…