Python进阶————面向对象高级

devtools/2024/11/20 0:22:57/

面向对象高级

  • 前言
  • 一、继承
    • 1.1. 单继承
    • 1.2. 多继承
    • 1.3. 方法重写
    • 1.4. 子类调用父类方法
      • 1.4.1 父类名.父类方法名()
      • 1.4.2 super().父类方法名()
    • 1.5. 多层继承
  • 二、封装
    • 2.1. 私有属性
    • 2.2. 私有方法
  • 三、多态
    • 3.1. 多态的条件
    • 3.2. 多态的定义
  • 四、面向对象的其他特性
    • 4.1. 对象属性
    • 4.2. 类属性
    • 4.3. 类方法
    • 4.4. 静态方法
  • 总结


前言

  • 接下来我们学习Python中的继承、封装、多态以及其他特性

一、继承

  • 面向对象中的继承: 指的是多个类之间的所属关系,即子类默认继承父类的属性和方法。

1.1. 单继承

语法

python">class 类名(父类名):代码...

一个摊煎饼的老师傅[master],在煎饼果子界摸爬滚打多年,研发了一套精湛的摊煎饼技术, 师父要把这套技术传授给他的唯一的最得意的徒弟[apprentice]。[初始化、无参]

python"># 1.定义师傅类
class Master(object):# 1.1 属性def __init__(self):self.kofu = '[传统方法]'# 1.2 方法def make_cake(self):print(f'使用{self.kofu}摊煎饼')# 2.定义徒弟类  继承自Master
class Tudi(Master):pass# 3.实例化
xiaoming = Tudi()         
print(xiaoming.kofu)	  # [传统方法]
xiaoming.make_cake()	  # 使用[传统方法]摊煎饼

1.2. 多继承

  • 多继承就是:一个类同时继承了多个父类。

语法

python">class 类名(父类名1,父类名2,...):  # 多个父类代码...

小明是个爱学习的好孩子,想学习更多的摊煎饼果子技术,于是,在百度搜索到小吃学校[school],学习摊煎饼果子技术。

python"># 1.定义师傅类
class Master(object):# 1.1 属性def __init__(self):self.kofu = '[传统方法]'# 1.2 方法def make_cake(self):print(f'使用{self.kofu}摊煎饼')# 2.定义学校类
class School(object):# 1.1 属性def __init__(self):self.kofu = '[学院派方法煎饼]'# 1.2 方法def make_cake(self):print(f'使用{self.kofu}摊煎饼')# 3.定义徒弟类
class Tudi(School, Master):pass# 3.实例化
xiaoming = Tudi()
print(xiaoming.kofu)    # [学院派方法煎饼]
xiaoming.make_cake()    # 使用[学院派方法煎饼]摊煎饼

注意:当一个类有多个父类时,默认使用第一个父类的同名属性和方法。

1.3. 方法重写

  • 重写也叫作覆盖,就是当子类属性或方法与父类的属性或方法名字相同的时候,从父类继承下来的成员可以重新定义!

xiaoming 掌握了老师傅的技术后,自己潜心钻研出一套自己的独门配方的全新摊煎饼果子技术。

python"># 1.定义师傅类
class Master(object):# 1.1 属性def __init__(self):self.kofu = '[传统方法]'# 1.2 方法def make_cake(self):print(f'使用{self.kofu}摊煎饼')# 2.定义学校类
class School(object):# 1.1 属性def __init__(self):self.kofu = '[学院派方法]'# 1.2 方法def make_cake(self):print(f'使用{self.kofu}摊煎饼')# 3.定义徒弟类
class Tudi(School,Master):# 1.1 属性def __init__(self):self.kofu = '[xiaoming独创技术]'# 1.2 方法def make_cake(self):print(f'使用{self.kofu}摊煎饼')# 3.实例化
xiaoming = Tudi()
print(xiaoming.kofu)   # [xiaoming独创技术]
xiaoming.make_cake()   # 使用[xiaoming独创技术]摊煎饼

当父类中跟子类中有同名方法的时候,子类可以重写父类方法,子类对象调用该方法的时候,优先使用自己的,用完再用父类的

1.4. 子类调用父类方法

  • 子类中仍想要保留父类的行为,则需要在子类中调用父类方法可以使用的以下两个方法:

1.4.1 父类名.父类方法名()

使用该方法的时候父类名.父类方法名(self) 括号中必须要有self

很多顾客都希望能吃到徒弟做出的有自己独立品牌的煎饼果子,也有学校配方技术的煎饼果子味道。

python"># 1.定义师傅类
class Master(object):# 1.1 属性def __init__(self):self.kofu = '[传统方法]'# 1.2 方法def make_cake(self):print(f'使用{self.kofu}摊煎饼')# 2.定义学校类
class School(object):# 1.1 属性def __init__(self):self.kofu = '[学院派方法]'# 1.2 方法def make_cake(self):print(f'使用{self.kofu}摊煎饼')# 3.定义徒弟类
class Tudi(School,Master):# 1.1 属性def __init__(self):self.kofu = '[独创技术]'# 1.2 方法def make_cake(self):print(f'使用{self.kofu}摊煎饼')# 1.3 使用老师傅的技术def master_make_cake(self):Master.__init__(self)Master.make_cake(self)# 1.4 使用学校技术def school_make_cake(self):School.__init__(self)School.make_cake(self)# 3.实例化
xiaoming = Tudi()
print(xiaoming.kofu)         # [独创技术]
xiaoming.make_cake()	     # 使用[独创技术]摊煎饼
xiaoming.master_make_cake()  # 使用[传统方法]摊煎饼
xiaoming.school_make_cake()	 # 使用[学院派方法]摊煎饼

1.4.2 super().父类方法名()

使用该方法的时候,super后边必须加括号,且 父类方法名括号中啥也不能有

很多顾客都希望能吃到徒弟做出的有自己独立品牌的煎饼果子,也有学校配方技术的煎饼果子味道。

python"># 1.定义师傅类
class Master(object):# 1.1 属性def __init__(self):self.kofu = '[传统方法]'# 1.2 方法def make_cake(self):print(f'使用{self.kofu}摊煎饼')# 2.定义学校类
class School(object):# 1.1 属性def __init__(self):self.kofu = '[学院派方法]'# 1.2 方法def make_cake(self):print(f'使用{self.kofu}摊煎饼')# 3.定义徒弟类
class Tudi(School,Master):# 1.1 属性def __init__(self):self.kofu = '[独创技术]'# 1.2 方法def make_cake(self):print(f'使用{self.kofu}摊煎饼')# 1.3 使用老师傅的技术def master_make_cake(self):# 必须重新初始化父类的属性Master.__init__(self)super().make_cake()# 1.4 使用学校技术def school_make_cake(self):# 必须重新初始化父类的属性School.__init__(self)super().make_cake()# 3.实例化
xiaoming = Tudi()
print(xiaoming.kofu)         # [独创技术]
xiaoming.make_cake()	     # 使用[独创技术]摊煎饼
xiaoming.master_make_cake()  # 使用[传统方法]摊煎饼
xiaoming.school_make_cake()	 # 使用[学院派方法]摊煎饼

1.5. 多层继承

python">"""
多层继承介绍:概述:实际开发中, 类与类之间是可以多层继承的, 例如: 类A继承类B, 类B继承类C, 这就是: 多层继承.例如:A => 继承B => 继承C => 继承object
"""

N年后,小明老了,想要把“有自己的独立品牌,也有学院配方技术的煎饼果子味道”的所有技术传授给自己的徒弟。

python"># 1. 创建1个师傅类, 充当父类.
class Master(object):# 1.1 定义父类的 属性.def __init__(self):self.kongfu = '[古法煎饼果子配方]'# 1.2 定义父类的 行为, 表示: 摊煎饼.def make_cake(self):print(f'使用 {self.kongfu} 制作煎饼果子')# 2. 创建1个师傅类, 充当父类.
class School(object):# 2.1 定义父类的 属性.def __init__(self):self.kongfu = '[学院派煎饼果子配方]'# 2.2 定义父类的 行为, 表示: 摊煎饼.def make_cake(self):print(f'使用 {self.kongfu} 制作煎饼果子')# 3. 定义徒弟类, 继承自师傅类.
class Prentice(School, Master):# 3.1 定义本类(子类)的 属性.def __init__(self):self.kongfu = '[独创煎饼果子配方]'# 3.2 定义本类(子类)的 行为, 表示: 摊煎饼.def make_cake(self):        # 子类出现和父类重名且一模一样的函数, 称之为: 方法重写.print(f'使用 {self.kongfu} 制作煎饼果子')# 3.3 定义函数 make_master_cake(), 表示: 古法摊煎饼果子配方.def make_master_cake(self):# 前提(细节): 需要重新初始化一下父类的 属性.Master.__init__(self)# 调用Master#make_cake()Master.make_cake(self)# 3.4 定义函数 make_school_cake(), 表示: 黑马AI摊煎饼果子配方.def make_school_cake(self):# 前提(细节): 需要重新初始化一下父类的 属性.School.__init__(self)# 调用School#make_cake()School.make_cake(self)# 4. 定义徒孙类, 继承: 徒弟类.
class TuSun(Prentice):      # 继承关系:  TuSun => Prentice => School, Master => objectpass# 在main函数中测试
if __name__ == '__main__':# 4. 创建 徒孙类 的对象ts = TuSun()# 5. 调用父类的成员.print(f'属性: {ts.kongfu}')  # 属性: [独创煎饼果子配方]ts.make_cake()               # 使用 [独创煎饼果子配方] 制作煎饼果子ts.make_master_cake()        # 使用 [古法煎饼果子配方] 制作煎饼果子ts.make_school_cake()        # 使用 [学院派煎饼果子配方] 制作煎饼果子

二、封装

  • 在软件编程中,将属性和方法书写到类的里面的操作即为封装,封装可以为属性和方法添加私有权限。

2.1. 私有属性

  • 在Python中,可以为属性设置私有权限,即设置某个属性不继承给子类。

  • 设置私有权限的方式:在属性名前面加上两个下划线 __,

格式

python">"""
self.__属性名当设置了属性为私有属性后,那么我们只能在本类中通过self. 的方法来访问该属性如果在类外想要访问私有属性的话,必须通过公共方法才行
"""

代码如下(示例):

python">class Prentice(object):# 1.1 属性def __init__(self):self.kongfu = '[独创的煎饼果子配方]'# 私有的属性.self.__money = 500000         # 这个才是私有化的写法.# 1.2 对外提供公共的访问方式, 可以实现: 获取私有的变量, 以及给变量设置值.# 获取值.def get_money(self):return self.__money# 设置值def set_money(self, money):# 可以在这里对 money属性做判断, 但是没必要. 因为Python属于后端代码, 这里的钱肯定是前端传过来的, 而传过来的数据已经经过了前端的校验.# 换言之, 这里如果校验就属于 二次校验了.  实际开发中, 重要字段会做二次校验, 否者可以不做校验.# if money > 0:#     self.__money = money# else:#     self.__money = 0self.__money = money# 2. 定义徒孙类, 继承自徒弟类.
class TuSun(Prentice):pass# 在main函数中测试调用
if __name__ == '__main__':# 3. 创建徒孙类对象.ts = TuSun()# 4. 尝试访问父类的成员.# 父类的公共的 属性print(f'父类的属性: {ts.kongfu}') # 父类的属性: [独创的煎饼果子配方]print(f'父类的私有属性, 通过 公共的方式访问: {ts.get_money()}')  # 父类的私有属性, 通过 公共的方式访问: 500000# 通过父类的公共方式, 修改 父类的私有属性.ts.set_money(10)print(f'父类的私有属性, 通过 公共的方式访问: {ts.get_money()}')  # 父类的私有属性, 通过 公共的方式访问: 10

2.2. 私有方法

  • 在Python中,可以为方法设置私有权限,即设置某个方法不继承给子类。
  • 设置私有权限的方式:在方法名前面加上两个下划线 __

格式

python">"""
def __方法名(self):...当设置了属性为私有属性后,那么我们只能在本类中通过self. 的方法来访问该属性如果在类外想要访问私有属性的话,必须通过公共方法才行
"""

小明把煎饼果子技术传承给徒弟的同时,不想把自己的独创配方制作过程继承给徒弟,这时就要为制作独创配方这个方法设置私有权限。

python"># 1. 定义徒弟类, 有自己的属性 和 行为.
class Prentice(object):# 1.1 属性def __init__(self):self.kongfu = '[独创的煎饼果子配方]'# 私有的属性.# self.__money__ = 500000     # 这个不是私有, 就是变量名叫: __money__self.__money = 500000         # 这个才是私有化的写法.# 1.2 对外提供公共的访问方式, 可以实现: 获取私有的变量, 以及给变量设置值.# 获取值.def get_money(self):return self.__money# 设置值def set_money(self, money):self.__money = money# 1.3 行为def __make_cake(self):print(f'采用 {self.kongfu} 制作煎饼果子!')# 验证: 私有成员, 在本类中是可以直接访问的.print(f'私房钱为: {self.__money}')# 针对于父类的私有方法, 提供公共的访问方式(在其内部调用 私有化的方法即可)def my_make(self):# 调用私有化方法 __make_cake()self.__make_cake()# 2. 定义徒孙类, 继承自徒弟类.
class TuSun(Prentice):pass# 在main函数中测试调用
if __name__ == '__main__':# 3. 创建徒孙类对象.ts = TuSun()# 4. 尝试访问父类的成员.# 4.1 父类的 私有的 属性.print(f'父类的私有属性, 通过 公共的方式访问: {ts.get_money()}')# 通过父类的公共方式, 修改 父类的私有属性.ts.set_money(10)print(f'父类的私有属性, 通过 公共的方式访问: {ts.get_money()}')print('-' * 21)# 4.2 父类的 私有的 方法(行为).# ts.__make_cake()        # AttributeError, 父类私有成员(方法), 子类无法直接访问.# ts.make_cake()ts.my_make()

三、多态

3.1. 多态的条件

python">"""
1、有继承 (定义父类、定义子类,子类继承父类)
2、函数重写 (子类重写父类的函数)
3、父类引用指向子类对象 (子类对象传给父类对象调用者)
"""

3.2. 多态的定义

  • 多态,指的是:多种状态。比如:同样一个函数在不同的场景下有不同的状态。

代码演示如下

python">class Animal(object):def speak(self):print("animal 嗯嗯 ")passclass Dog(Animal):def speak(self):print('汪汪汪')class Cat(Animal):def speak(self):print('喵喵喵')# def make_noise(animal):
def make_noise(animal:Animal):animal.speak()if __name__ == '__main__':mydog = Dog()mycat = Cat()animal =  Animal()make_noise(mydog)     # 汪汪汪# make_noise(mycat)   # 喵喵喵# make_noise(animal)   # animal 嗯嗯pass

四、面向对象的其他特性

4.1. 对象属性

python">"""
对于属性,调用时有两种方式,如下:1、 self.对象属性名  # 类内部2、 对象名.对象属性名   # 类外部
"""

4.2. 类属性

python">"""
所谓类属性,指的就是类所拥有的属性,它被共享于整个类中(即都可以直接调用),格式如下:1、 类名.类属性名    # 推荐使用2、 对象名.类属性名"""

4.3. 类方法

  • 类方法,指的是类所拥有的方法,并需要使用[装饰器]@classmethod来标识其为类方法

同时一定要注意的是对于类方法的第一个参数必须是类对象,通常以cls作为第一个参数名,格式如下:

python">@classmethod
def 类方法名(cls):...# 调用时,格式如下:1、 类名.类方法名()    # 推荐使用2、 对象名.类方法名()

类方法一般和类属性配合使用,尤其是使用私有类属性时。

4.4. 静态方法

静态方法需要通过装饰器@staticmethod来来标识其为静态方法,且静态方法不需要多定义参数,格式如下:

python">@staticmethod
def 静态方法名():...# 调用时,格式如下:1、 类名.静态方法名    # 推荐使用2、 对象名.静态方法名

功能不变的时候使用静态方法


总结

  • 以上就是面向对象的封装、继承、多态的定义与代码演示。

http://www.ppmy.cn/devtools/104755.html

相关文章

【C++ Primer Plus习题】8.6

问题: 解答: #include <iostream> using namespace std;template <typename T> T maxn(T arr[], int len)//通用 {T max 0;for (int i 0; i < len; i){if (max < arr[i]){max arr[i];}}return max; }template<> const char* maxn<const char*&g…

基于 ASP.NET的教材管理信息系统的设计与实现(最新定制开发,阿龙原创设计)✅

博主介绍&#xff1a; ✌我是阿龙&#xff0c;一名专注于Java技术领域的程序员&#xff0c;全网拥有10W粉丝。作为CSDN特邀作者、博客专家、新星计划导师&#xff0c;我在计算机毕业设计开发方面积累了丰富的经验。同时&#xff0c;我也是掘金、华为云、阿里云、InfoQ等平台…

谷歌浏览器http自动跳转https问题

1.地址栏输入&#xff1a; chrome://net-internals/#hsts 2.找到底部Delete domain security policies一栏&#xff0c;输入想处理的域名&#xff0c;点击delete。 3.搞定了&#xff0c;再次访问http域名不再自动跳转https了。

【C++】智能指针——auto_ptr,unique_ptr,shared_ptr

目录 auto_ptr unique_ptr shared_ptr 并发问题 循环引用问题 个人主页&#xff1a;传送门——>东洛的克莱斯韦克 智能指针的原理&#xff1a;传送门——>智能指针的原理 auto_ptr 使用方法参考官方文档 传送门——>auto_ptr文档 auto_ptr并不是一个优秀的智能…

split对大文件(tar/tar.gz)文件进行分片及合并

文章目录 1、tar文件指定大小分片2、合并分片文件并解压 1、tar文件指定大小分片 split -b 4000M -d -a 3 cm-11.tar.gz cm-11.tar.gz.使用split命令&#xff0c;-b 4000M 表示设置每个分割包的大小&#xff0c;单位还是可以k -d "参数指定生成的分割包后缀为数字的形式 …

Python 生成随机的国内 ip

示例代码&#xff1a; import randomdef generate_random_cn_ip():# 中国大陆IP范围start_ip "36.54.0.0"end_ip "123.255.255.254"# 将IP地址转换为整数start_ip_num int(start_ip.replace(".", ""))end_ip_num int(end_ip.rep…

如何防范ddos 攻击

防护DDoS&#xff08;分布式拒绝服务&#xff09;攻击是一个复杂且多方面的任务&#xff0c;需要综合考虑网络架构、硬件设备、软件配置以及安全策略等多个方面。以下是一些主要的防护DDoS攻击的方法&#xff1a; 1. 流量清洗&#xff08;Traffic Scrubbing&#xff09; 定义…

.net dataexcel winform控件 更新 日志

增加 列宽度调整时动态显示列象素大小 更改列的宽度可以使用 column.Width属性进行修改