《C程序设计教程第4章.ppt》由会员分享,可在线阅读,更多相关《C程序设计教程第4章.ppt(62页珍藏版)》请在三一办公上搜索。
1、2023/6/27,1,C#程序设计经典教程,第四章 面向对象程序设计入门,1.总体要求理解面向对象的基本概念,正确区分类和对象,对象的声明和对象的创建。掌握类的定义与使用方法,正确定义类的数据成员、属性和方法。理解类的可访问性、正确使用访问修饰符控制对类成员的访问。掌握类的方法的定义、调用与重载,理解方法的参数传递的工作机制。理解值类型和引用类型的区别。理解构造函数与析构函数的作用,掌握其使用方法。方法的重载和参数传递,2.相关知识点熟悉C#中数据类型、表达式、运算符、常量与变量等基础知识。熟悉C#中数据类型转换。3.学习重点C#中类的定义、类的数据成员、属性和方法类的构造函数方法的重载和参
2、数传递,第四章 面向对象程序设计入门,4.学习难点类和对象的关系方法的重载和参数传递值类型和引用类型及对象的生命周期,第四章 面向对象程序设计入门,第四章 面向对象程序设计入门,主要内容4.1 面向对象的基本概念 4.2 类的定义4.3 类的方法4.4 构造函数,2023/6/27,6,4.1 面向对象的基本概念,4.1.1 对象4.1.2 事件与方法4.1.3 类与对象4.1.4 抽象、封装、继承与多态,返回,2023/6/27,7,4.1.1 对象,客观世界中任何一个事物都可以看成一个对象(object),对象可以是自然物体(如汽车、房屋、狗),也可以是社会生活中一种逻辑结构(如班级、部门
3、、组织),甚至一篇文章、一个图形、一项计划等都可以视作对象。对象是构成系统的基本单位,在实际社会生活中,人们都是在不同的对象中活动的。任何一个对象都应当具有这两个要素,即属性(attribute)和行为(behavior),一个对象往往由一组属性和一组行为构成,一辆汽车是一个对象,它的属性是生产厂家、品牌、型号、颜色、价格等,它是行为是它的功能,如发动、停止、加速等,一般来说,凡是具备属性和行为这两个要素的,都可以作为对象,,4.1.1 对象,对象是问题域中某些事物的一个抽象,反映事物在系统中需要保存的必要信息和发挥的作用,是包含一些特殊属性(数据)和服务(行为方法)的封装实体。具体来说,他应
4、有唯一的名称,有一系列状态(表示为数据),有表示对象行为的一系列行为(方法),简言之:对象=属性+行为(方法、操作),4.1.2 事件与方法,事件(Event)又称为消息(Message),表示向对象发出的服务请求。方法(Method)表示对象能完成的服务或执行的操作功能。在一个系统中的多个对象之间通过一定的渠道相互联系,要使某一个对象实现某一种行为或操作,应当向他传送相应的消息。例如想让汽车行驶,必须由人去踩油门,向汽车发出相应的信号。对象之间就是这样通过发送和接收消息互相联系的。在面向对象的概念中,一个对象可以有多个方法,提供多种服务,完成多种操作功能。但这些方法只有在另外一个对象向他发出
5、请求之后(发生事件)才会被执行。,4.1.3 类与对象,普通逻辑意义上的类是现实世界中各种实体的抽象概念,而对象是现实生活中的一个个实体,例如,在现实世界中大量具体一辆辆汽车、摩托车、自行车等实体是对象,而交通工具则是这些对象的抽象,交通工具就是一个类。在面向对象的概念中,类(Class)表示具有相同属性和行为的一组对象的集合,为该类的所有对象提供统一的抽象描述。类是对相似对象的抽象,而对象是该类的一个特例,类与对象的关系是抽象与具体的关系。,4.1.4 抽象、封装、继承与多态,面向对象的最基本的特征是抽象性、封装性、继承性和多态性。1抽象抽象(abstraction)是处理事物复杂性的方法,
6、只关注与当前目标有关的方面,而忽略与当前目标无关的那些方面,例如在学生成绩管理中,张三、李四、王五作为学生,我们只关心他们和成绩管理有关的属性和行为,如学号、姓名、成绩、专业等特性。抽象的过程是将有关事物的共性归纳、集中的过程,例如凡是有轮子、能滚动并前进的陆地交通工具统称为“车子”,把其中用汽油发动机驱动的抽象为“汽车”,把用马拉的抽象为“马车”。,4.1.4 抽象、封装、继承与多态,抽象能表示同一类事物的本质,如果你会使用自己家里的电视机,在别人家里看到即便是不同的牌子的电视机,你也能对它进行操作。因这它具有所有电视机所共有的特征,而C#中的数据类型就是对一系列具体的数的抽象,例如:int
7、是对所有整数的抽象,double是对所有双精度浮点型数的抽象。,2封装和信息隐藏封装(encapsulation)有两个方面的含义:一是将有关的数据和操作代码封装在一个对象中,形成一个基本单位,各个对象之间相对独立,互不干扰。二是将对象中某些部份对外隐藏,即隐藏其内部细节,只留下少量接口,以便与外界联系,接收外界的消息。这种对外界隐藏的做法称为信息隐藏(information hiding)。信息隐藏还有利于数据安全,防止无关的人了解和修改数据。,4.1.4 抽象、封装、继承与多态,封装把对象的全部属性和全部行为结合在一起形成一个不可分割的独立单位。而通过信息隐蔽技术,用户只能见到对象封装界面
8、上的信息,对象内部对用户是隐蔽的。例如,一台电视机就是一个封装体。从设计者的角度来讲,不仅需要考虑内部的各种元器件,还要考虑主机板、显像管等元器件的连接与组装;从使用者的角度来讲,只关心其型号、颜色、重量等属性,只关心电源开关按钮、音量开关、调频按钮、视频输入输出接口等用起来是否方便,根本不用关心其内部构造。因此,封装的目的在于将对象的使用者与设计者分开,使用者不必了解对象行为的具体实现,只需要用设计者提供的消息接口来访问该对象。,4.1.4 抽象、封装、继承与多态,4.1.4 抽象、封装、继承与多态,3继承汽车制造厂要生产新型号的汽车,如果全部从头开始设计,将耗费大力的人力、物力和财力。但如
9、果选择已有的某一型号的汽车为基础,再增加一些新的功能,就能快速研发出新型号的汽车。这是提高生产效率的常用方法。如果在软件开发中已建立了一个名为A的类,又想建立一个名为B的类,而后者与前者内容基本相同,只是在前者基础上增加一些新的属性和行为,显然不必再从头设计一个新类,只需在A类的基础上增加一些新的内容即可,而B类的对象拥有A类的全部属性与方法,称作B类对A类的继承,在B类中不必重新定义已在A类中定义过的属性和方法,这种特性在面向对象中称作对象的继承性。继承在C#中称为派生,其中,A类称为基类或父类,B类称为派生类或子类。,4.1.4 抽象、封装、继承与多态,继承机制的优势在于降低了软件开发的复
10、杂性和费用,使软件系统易于扩充,大大缩短了软件开发周期,对于大型软件的开发具有重要的意义。,4.1.4 抽象、封装、继承与多态,4多态多态性(polymorphism)是指在基类中定义的属性或方法被派生类继承后,可以具有不同的数据类型或表现出不同的行为。其对象对同一消息会做出不同的响应,如张三、李四和王五是分别是属于三个班的三个学生,在听到上课铃声后,他们会分别走进3个不同的教室,同样,“启动”是所有交通工具都具有的操作,但不同的具体交通工具其“启动”操作的具体实现是不同的,如汽车的启动是“发动机点火,启动引擎”,启动轮船时要“起锚”,气球飞艇启动是“充气,解缆”,为了实现多态性,需要在派生类
11、中更改从基类中自动继承来的数据类型或方法。这种为了替换基类的部分内容而在派生类中重新进行定义的操作,在面向对象的概念中称为覆盖。这样一来,不同类的对象可以响应同名的消息(方法)来完成特定的功能,但其具体的实现方法却可以不同。,4.2 类的定义,4.2.1 类的声明和实例化 4.2.2 类的数据成员和属性4.2.3 类的可访问性4.2.4 值类型与引用类型,返回,4.2.1 类的声明和实例化,1类的声明类的声明一般形式如下:访问修饰符 class 类名:基类 类的成员;其中,访问修饰符用来限制类的作用范围或访问级别,可省略;类名是一个合法的C#标识符,推荐使用Pascal命名规范,Pascal命
12、名规范要求名称的每个单词的首字母要大写;基类表明所定义的类是一个派生类,可省略;类的成员放在花括号中,构成类的主体,用来定义类的属性和行为。类的成员包括常量、字段、属性、索引器、方法、事件、构造函数等。,4.2.1 类的声明和实例化,2类的实例定义类之后,可以用定义的类声明对象,然后再通过这个对象来访问其数据或调用其方法。(1)对象的声明与创建。声明对象的格式与声明基本数据类型的格式相同,其语法格式为:类名 对象名例如:Student stu1;/声明一个Student对象stu1,4.2.1 类的声明和实例化,但是,对象声明后需要用“new”关键字将对象实例化,这样才能为对象在内存中分配保存
13、数据空间,实例化的语法格式为:对象名=new 类名();例如:stu1=new Student();/为stu1分配内存空间也可以在声明对象同时实例化对象。语法格式为:类名 对象名=new 类名();例如:Student stu2=new Student();/声明同时创建对象,4.2.1 类的声明和实例化,(2)类成员的访问。类成员有两种访问方式:一种是在类的内部访问,另一种是在类的外部访问。在类的内部访问类的成员,表示一个类成员要使用当前类中的其他成员,可以直接使用成员名称,有时为了避免引起混淆,也可采用如下形式:this.类成员其中,this表示当前对象,是C#的关键字。,4.2.1 类
14、的声明和实例化,【实例4-1】定义Student类并实例化类的对象。,4.2.2 类的数据成员和属性,类的成员包括类的常量、字段、属性、索引器、方法、事件、构造函数等,其中,常量、字段和属性都是与类的数据有关的成员。1常量常量的值是固定不变的。类的常量成员是一种符号常量,符号常量是由用户根据需要自行创建的常量,在程序设计过程中可能需要反复使用到某个数据,如圆周率3.1415926,如果在代码中反复书写,不仅麻烦而且容易出现书名错误,此时,可考虑将其声明为一个符号常量,用户定义符号常量使用const关键字,在定义时,必须指定名称和值,其一般形式如下:访问修饰符 const 数据类型 常量名=常量
15、的值;,4.2.2 类的数据成员和属性,2字段字段表示类的成员变量,字段的值代表某个对象的数据状态。不同的对象,数据状态不同,意味着各字段的值也不同。声明字段的方法与定义普通变量的方法相同,其一般格式如下:访问修饰符 数据类型 字段名;其中,访问修饰符用来控制字段的访问级别,可省略。例如:public double radius;,4.2.2 类的数据成员和属性,3属性字段和常量描述了类的数据,当这些数据允许外界访问时,可以使用访问修饰符public,不允许外界访问时,可以使用private或protected 定义属性的一般形式如下:访问修饰符 数据类型 属性名 get/获取属性的代码,用r
16、eturn 返回值 set/设置属性的代码,用value赋值,4.2.2 类的数据成员和属性,【实例4-2】定义类的数据成员及属性。,4.2.3 类的可访问性,为了控制类和类成员的作用范围或访问级别,C#提供了访问修饰符,用于限制对类和类成员的访问属性。这些访问修饰符包括public、private、internal、protected、protected internal在使用访问修饰符来定义命名空间、结构和类及其成员时,要注意以下几点。(1)一个成员或类型只能有一个访问修饰符,使用 protected internal 组合时除外。(2)命名空间上不允许使用访问修饰符,命名空间没有访问限制
17、。(3)如果未指定访问修饰符,则使用默认的可访问性,类的成员默认为private,如实例4-2中的pi没有指定访问修饰符,默认为private。(4)第一级类型(指不嵌套在其他类型中的类型)的可访问性只能是internal或public,默认可访问性是internal,如实例4-2中的Circle类没有指定访问修饰符,默认为internal。,4.2.4 值类型与引用类型,C#将数据类型分为值类型(value type)和引用类型(reference type)。1值类型值类型变量直接包含其本身的数据,前面提到的简单类型(int、bool、char、float、double、decimal)、
18、结构类型(struct)、枚举类型(enum)等都是值类型。在定义一个值类型变量后,将直接为该变量分配空间,可以直接赋值和使用。,4.2.4 值类型与引用类型,引用类型与值类型不同,引用类型变量本身并不包含数据,只是存储对数据的引用,数据保存在其他位置,数组、字符串、类和后面要介绍的接口、委托等都属于引用类型。引用型变量在定义时并不会分配空间,只是在对其实例化后才真正的分配存储空间。,4.2.4 值类型与引用类型,3装箱和折箱对于值类型来说,可以通过隐式转换方法或显示转换方法进行数据转换;对于引用类型来说,C#同样允许将任何类型的数据转换为对象,或者将任何类型的对象转换为与之兼容的数据类型。C
19、#把值类型转换为对象的操作称为装箱,而把对象转换为兼容的值类型的操作称为拆箱。C#的这种装箱与拆箱操作类似于收发邮政包裹,发送包裹之前先装箱打包,收到包裹后再拆箱解包。装箱意味着把一个值类型的数据转换为一个对象类型的数据,装箱过程是隐式转换过程,由系统自动完成,C#中Object类是所有类的最终基类,因此,可以将一个值类型变量直接赋值给Object对象,4.3 类的方法,4.3.1 方法的声明与调用 4.3.2 方法的参数传递4.3.3 方法的重载,返回,4.3.1 方法的声明与调用,方法的使用分声明与调用两个环节。1方法的声明声明方法的一般形式如下:访问修饰符 返回值类型 方法名(参数列表)
20、语句;return 返回值;,4.3.1 方法的声明与调用,(1)访问修饰符控制方法的访问级别,可用于方法的修饰符包括public、protected、private和internal等;访问修饰符是可选的,默认情况下为 private(2)方法的返回类型用于指定由该方法计算和返回的值的类型,可以是任何合法的数据类型,包括值类型和引用类型,如果一个方法不返回一个值,则返回值类型使用void关键字来表示;(3)方法名必须符合C#的命名规范,与变量名的命名规则相同;,4.3.1 方法的声明与调用,(4)参数列表是方法可以接受的输入数据,当方法不需要参数时,可省略参数列表,但不能省略圆括号;当参数不
21、止一个时,需要使用逗号分隔,同时每一个参数都必须声明数据类型,即使这些参数的数据类型相同也不例外;(5)花括号中的内容为方法的主体,由若干条语句组成,每一条语句都必须使用分号结尾。当方法结束时如果需要返回操作结果,则使用return语句返回,并且返回的值的类型要与返回值的类型相匹配。如果使用void标记方法为无返回值的方法,可省略return语句。,4.3.1 方法的声明与调用,2方法的调用一个方法一旦在某个类中声明,就可由其他方法调用,调用者既可以是同一个类中的方法,也可以是其他类中的方法。如果调用者是同一个类的方法,则可以直接调用,如果调用者是其他类中的方法,则需要通过类的实例来引用,但静
22、态方法例外,静态方法通过类名直接调用(1)在方法声明的类定义中调用该方法。其语法格式为:方法名(参数列表)(2)在方法声明的类定义外部调用该方法,需要通过类声明的对象调用该方法,其格式为:对象名.方法名(参数列表),4.3.2 方法的参数传递,在声明方法时,所定义的参数是形式参数(简称形参),这些参数的值由调用方负责为其传递,调用方传递的是实际数据,称为实际参数(简称实参),调用方必须严格按照被调用的方法所定义的参数类型和顺序指定实参。在调用方法时,参数传递就是将实参传递给形参的过程。方法的参数传递按性质可分为按值传递与按引用传递。,4.3.2 方法的参数传递,1按值传参按值传参时,把实参变量
23、的值赋给相对应的形参变量,即被调用的方法所接收到的只是实参数据值的一个副本。当在方法内部更改了形参变量的数据值时,不会影响实参变量的值,即实参变量和形参变量是两个不相同的变量,他们具有各自的内存地址和数据值。因此,实参变量的值传递给形参变量时是一种单向值传递。值类型的参数在传递时默认为按值传参。string和object虽然是引用型数据,但从表现形式来看,其具有按值传参的效果。,4.3.2 方法的参数传递,【实例4-3】用值传参进行参数值交换,4.3.2 方法的参数传递,2按引用传参方法只能返回一个值,但在实际应用中常常需要方法能够返回多个值或修改传入的参数值并返回,如果需要完成以上任务,只用
24、return语句是无法做到的,这时可以使用按引用传递参数的方式来实现。调用方向方法传递引用型参数时,调用方将把实参变量的引用赋给相对应的形参变量。实参变量的引用代表数据值的内存地址,因此,形参变量和实参变量将指向同一个引用。如果在方法内部更改了形参变量所引用的数据值,则同时也修改了实参变量所引用的数据值。当值类型和string类型参数要按引用方式传参时,可以通过ref关键字来声明引用参数,无论是形参还是实参,只要希望传递数据的引用,就必须添加ref关键字。,4.3.2 方法的参数传递,【实例4-4】用引用传参进行参数值交换。(1)将【实例4-3】Swap方法声明改为引用型参数:public s
25、tring Swap(ref int x,ref int y)(2)将【实例4-3】Swap方法调用改为引用型传参:lblShow.Text=s.Swap(ref a,ref b);,4.3.2 方法的参数传递,3输出参数方法中的return语句只能返回一个运算结果,虽然也可以使用引用型参数返回计算结果,但用ref修饰的参数在传参前要求先初始化实参。但有时候参数在传参前无法确定其值,其值应由方法调用结束后返回,所在在传参前确定其值是没有意义的。这时可以使用输出参数,输出参数不需要对实参进行初始化,它专门用于把方法中的数据通过形参返回给实参,但不会将实参的值传递给形参。一个方法中可允许有多个输出
26、参数。C#通过out关键字来声明输出参数,无论是形参还是实参,只要是输出参数,都必须添加out关键字。,4.3.2 方法的参数传递,【实例4-5】用输出参数求文件路径中的目录和文件名。,4.3.2 方法的参数传递,4引用类型数据的传参引用类型参数总是按引用传递的,所以引用类型参数传递不需要使用ref或out关键字(string除外),引用类型参数的传递,实际上是将实参对数据的引用复制给了形参。所以形参与实参共同指向同一个内存区域。【实例4-6】用引用类型数据的传参修改对象值。,4.3.2 方法的参数传递,5数组型参数数组也是引用类型数据,把数组作为参数传递时,也是引用传参。但把数组作为参数,有
27、两种使用形式:一种是在形参数组前不添加params修饰符,另一种是在形参数组前添加params修饰符。不添加params修饰符时,所对应的实参必须是一个数组名;添加params修饰符时,所对应的实参可以是数组名,也可以是数组元素值的列表,此时,系统将自动把各种元素值组织到一个数组中。无论采用哪一种形式,形参数组都不能定义数组的长度。,4.3.2 方法的参数传递,【实例4-7】使用不添加params和添加params修改符的数组传参求数组中的最大值。,4.3.3 方法的重载,在编程时,一般是一个方法对应一种功能,但有时需要实现同一类功能,只是有些细节不同。例如希望从几个数中找出其中的最大者,而每
28、次数据个数或类型不同,如2个整数,2个双精度数、3个整数、或一个整型数组做为参数。这时,我们可以设计出4个不同名的方法,其形式为:public int MaxIntTwo(int a,int b)public double MaxDouble(double a,double b)public int MaxIntThree(int a,int b,int c)public int MaxArray(int a)这时,程序者需要以不同的方法名来命名这些功能类似的方法,而调用者更需记住不同的方法名,不是很方便,在C#中,允许用同一方法名定义多个方法,这些方法的参数个数或参数类型不同,这就是方法的重
29、载(function overloading)。,4.3.3 方法的重载,方法重载有两点要求:(1)重载的方法名称必须相同;(2)重载方法的形参个数或类型必须不同,否则将出现一个“已定义了一个具有相同参数类型的成员”如果要完成上例相似功能的4个方法,重载方法如下:public int Max(int a,int b)public double Max(double a,double b)public int Max(int a,int b,int c)public int Max(int a)在调用具有重载的方法时,系统会根据参数的类型或个数确定最匹配的方法被调用。,4.3.3 方法的重载,【
30、实例4-8】利用方法重载实现2个整数,2个双精度数、3个整数中求最大值。,4.4 构造函数,4.4.1 构造函数的声明和调用 4.4.2 构造函数的重载 4.4.3 构造函数与只读字段 4.4.4 对象的生命周期和析构函数,返回,4.4.1 构造函数的声明和调用,构造函数是类中的一种特殊的方法,其一般形式如下:public 构造函数名(参数列表)语句;和普通方法相比,构造函数有两个特别要求,一是构造函数的名称必须和类名相同,二是构造函数不允许有返回类型(包括void类型)。其中,构造函数的参数列表可省略,也可以不包含任何语句。不包含任何参数和语句的构造函数称为默认构造函数。如果没有定义构造函数
31、,编译器将自动生成默认构造函数由,默认构造函数的形式如下:public 构造函数名(),4.4.2 构造函数的重载,构造函数重载与方法一样可以重载。在一个类中,可以定义多个构造函数,以便对对象提供不同的初始化方法,以满足创建对象时的不同需要。例如,在创建一个Student对象时,只想指定name的值,而age为默认的20。可以声明一个如下所示的构造函数:public Student(string name)this.name=name;this.age=20;该构造函数和以上面的public Student(string name,int age)构造函数相比,参数的个数不同,是一个合法的构造
32、函数。此时,可以声明一个只有一个实参的对象:Student stu=new Student(郭靖);,4.4.2 构造函数的重载,【实例4-9】利用构造函数重载实现不同对象实例化。,4.4.3 构造函数与只读字段,C#中类的字段成员可通过关键字readonly设置为只读字段,对于标记为只读的字段来说,只有在声明时为它赋值,或者在对象初始化时赋值。在声明时为只读字段赋值与声明常量没有区别,在对象初始化时为只读字段赋值需要使用构造函数实现。,4.4.3 构造函数与只读字段,【实例4-10】公园门票调价问题。,4.4.4 对象的生命周期和析构函数,C#程序中,一个对象是类的一个实例,实际上就是一个引
33、用型的变量,在程序运行过程中,它需要占用一定的内存空间,.NET 的公共语言运行时负责为其分配内存。当程序运行结束后,需要回收它所占用的内存空间。正如前面的介绍一样,.NET的公共语言运行时把值类型变量和引用型变量放在不同的内存区域中管理。值类型变量使用“栈”来管理,栈是一种按照“先进后出”方式存取的内存区域。当方法被调用时,方法内的值类型变量(包括形参变量)自动获得内存,当方法调用结束时,这些变量所占用的内存将自动释放。,4.4.4 对象的生命周期和析构函数,引用型变量使用“堆”来管理,堆是分配对象时所使用的内存区域。在方法调用过程中,一旦使用运算符new创建了对象,.NET的公共语言运行时
34、立即为该对象从堆中分配内存。当方法调用结束时,对象所占用的内存并不会自动从堆中释放。在.NET中,对象所占用的内存只能由.NET的公共语言运行时的垃圾回收器来回收,垃圾回收器没有预定的工作模式,其工作时间间隔是不可预知的,垃圾回收器的优化引擎能根据分配情况确定回收的最佳时机。,4.4.4 对象的生命周期和析构函数,一个对象的生命周期可分为以下几个阶段:(1)使用new运算符创建对象并要求获得内存;(2)对象初始化,包括对象的数据成员的初始化;(3)使用对象,包括访问对象的数据成员、调用对象的方法成员;(4)对象释放前释放对象所占用的资源,如关闭磁盘文件、网络连接等;(5)释放对象,回收内存(由
35、垃圾回收器自动完成)。其中,第2阶段可通过调用对象的构造函数来完成,第4阶段可通过析构函数来完成。,4.4.4 对象的生命周期和析构函数,C#类的析构函数具有如下特点:(1)不能在结构中定义析构函数,只能对类使用析构函数;(2)一个类只能有一个析构函数;(3)无法继承或重载析构函数;(4)析构函数既没有修饰符,也没有参数;(5)在析构函数被调用时,.NET的公共语言运行时自动添加对基类Object.Finalize方法的调用,以清理现场,因此在析构函数中不能包含对Object.Finalize方法的调用。,4.4.4 对象的生命周期和析构函数,析构函数的一般形式如下:函数名()语句;在默认情况下,编译器自动生成空的析构函数,因此C#不允许定义空的析构函数。其实析构函数性能较差,并不推荐值使用,如果需要尽快关闭和释放所占用的资源,应实现一个强制回收方法,一般称为close()或Dispose()。,2023/6/27,61,作业,1.书面作业(见本章教材)2.上机实验(见本章教材),2023/6/27,62,Class Over,Thank you!,
链接地址:https://www.31ppt.com/p-5336556.html