python中的类与对象(3)

目录

一. 类的多继承

二. 类的封装

三. 类的多态

四. 类与对象综合练习:校园管理系统


一. 类的多继承

在(2)第四节中我们介绍了什么是类的继承,在子类的括号里面写入要继承的父类名。上一节我们只在括号内写了一个父类名,但其实写入多个也是可以的,这说明它同时继承了多个父类,这就是多继承。给出一段代码示例如下:

class Shenxian:def fly(self):print("神仙在飞...")class Monkey:def eat_peach(self):print("猴子在偷吃仙桃...")class Sunhouzi(Shenxian, Monkey):def fangendou(self):print("孙悟空在翻跟斗...")S = Sunhouzi()
S.eat_peach()

这里Sunhouzi就同时继承了Shenxian和Monkey类,孙猴子也具有Shenxian和Monkey的方法。这段代码也印证了,如果子类没有的方法,就要去父类里面寻找同名方法。在多继承的条件下,如果多个父类都有同名方法,它首先去寻找哪个父类下的同名方法呢?

class Shenxian:def fly(self):print("神仙在飞...")def fight(self):print("神仙在打架...") class Monkey:def eat_peach(self):print("猴子在偷吃仙桃...")def fight(self):print("猴子在打架...")class Sunhouzi(Monkey, Shenxian):def fangendou(self):print("孙悟空在翻跟斗...")S = Sunhouzi()
S.fight()  # 猴子在打架...

可见它是按照括号内的顺序调用的,从左往右发现某父类有就直接调用该父类下的方法。

现在我们考虑更复杂的情况:把Monkey类下的fight下方法删除,同时写一个Organism类,让Shenxian和Monkey类都继承Organism类。此时相当于多重继承:Sunhouzi类继承了Shenxian和Monkey类,Shenxian和Monkey类又继承Organism类:

class Organism:def fight(self):print("群魔乱舞,各种生物都在打架...")class Shenxian(Organism):def fly(self):print("神仙在飞...")def fight(self):print("神仙在打架...") class Monkey(Organism):def eat_peach(self):print("猴子在偷吃仙桃...")class Sunhouzi(Shenxian, Monkey):def fangendou(self):print("孙悟空在翻跟斗...")S = Sunhouzi()
S.fight()  # 神仙在打架...

此时我们再让孙猴子打架,得到的输出是“神仙在打架...”

表面上来看,这类似于图的广度优先遍历(关于什么是图的广度优先遍历和深度优先遍历,可以参考数据结构博客:20.图的遍历-CSDN博客),然而实际上多重继承时确定子类继承哪一个父类方法用的是C3算法。C3算法实现了三种重要特性:

  • 保持继承拓扑图的一致性。
  • 保证局部优先原则(比如A继承C,C继承B,那么A读取父类方法,应该优先使用C的方法而不是B的方法)。
  • 保证单调性原则(即子类不改变父类的方法搜索顺序)。

我们并不去详细介绍C3算法(因为没有开发人员这么去干),只给出一段代码说明多重继承时既不满足广度优先遍历也不满足深度优先遍历:

class A:def test(self):print("from A")class B(A):passclass B2:def test(self):print("from B2")class C(A):def test(self):print("from C")class C2:def test(self):print("from C2")class D(B, B2):passclass E(C, C2):passclass F(D, E):passf1 = F()
f1.test()
# from C

用C3方法解析的顺序即方法解析顺序(Method Resolution Order,MRO),我们可以用mro()查看,这样我们不需要懂C3算法也可以知道它的方法解析顺序:

print(F.mro()) 
# [<class '__main__.F'>, <class '__main__.D'>, <class '__main__.B'>, <class '__main__.E'>, <class '__main__.C'>, <class '__main__.A'>, <class '__main__.B2'>, <class '__main__.C2'>, <class 'object'>]

二. 类的封装

封装可以被认为是一个保护屏障,防止该类的代码和数据被外部类定义的代码随机访问要访问该类的代码和数据,必须通过严格的接口控制。封装最主要的功能在于我们能修改自己的实现代码,而不用修改那些调用我们代码的程序片段。适当的封装可以让程式码更容易理解与维护,也加强了代码数据的安全性。封装的优点如下:

  1. 良好的封装能够减少耦合。
  2. 类内部的结构可以自由修改。
  3. 可以对成员变量进行更精确的控制。
  4. 隐藏信息,实现细节。

封装的原则是:

  1. 将不需要对外提供的内容都隐藏起来;
  2. 把属性都隐藏,提供公共方法对其访问。

以下是一个没有封装的例子,我们可以不打枪就使得人的血量发生减少:

class Person:def __init__(self, name, sex):self.name = nameself.sex = sexself.life_val = 100a = Person("zhangsan", "M")
a.life_val = 50
print(a.life_val)  # 50

为了解决这个问题我们需要对变量进行私有化,即加__,此时在外部是无法访问私有属性life_val的:

class Person:def __init__(self, name, sex):self.name = name  # 实例变量,成员变量self.sex = sexself.__life_val = 100  # 私有变量,私有属性,加__使得变量私有化a = Person("zhangsan", "M")
print(a.__life_val)  
# AttributeError: 'Person' object has no attribute '__life_val'

但是私有属性在类的内部是可以被访问的,如:

class Person:def __init__(self, name, sex):self.name = nameself.sex = sexself.__life_val = 100  def get_life_val(self):print("生命值还有:", self.__life_val)def got_attack(self):self.__life_val -= 20print("[%s]受到了攻击,掉了20滴血,现在生命值是[%s]" % (self.name, self.__life_val))a = Person("zhangsan", "M")
a.get_life_val()  # 生命值还有: 100
a.got_attack()  # [zhangsan]受到了攻击,掉了20滴血,现在生命值是[80]
a.get_life_val()  # 生命值还有: 80

如此我们就可以让life_val只读,而不能对其进行外部修改。

同理,我们也可以对方法进行封装,只在类的内部进行调用:

class Person:def __init__(self, name, sex):self.name = nameself.sex = sexself.__life_val = 100  def __breath(self):print(self.name,"在呼吸...")a = Person("zhangsan", "M")
a.__breath()  # AttributeError: 'Person' object has no attribute '__breath'

如果非要在外部进行访问,则需要写:实例对象._类名+方法名,相当于解封装:

class Person:def __init__(self, name, sex):self.name = name  # 实例变量,成员变量self.sex = sexself.__life_val = 100  # 私有变量,私有属性,加__使得变量私有化a = Person("zhangsan", "M")
print(a._Person__life_val)  # 100
# 如果直接写print(a.__life_val),则会报AttributeError

三. 类的多态

有时一个对象会有多种表现形式,比如网站页面有个button按钮,这个button的设计可以不一样(单选框、多选框、圆角的点击按钮、直角的点击按钮等),尽管长的不一样,但它们都有一个共同调用方式,就是onClick()方法。我们直要在页面上一点击就会触发这个方法。点完后有的按钮会变成选中状态、有的会提交表单、有的甚至会弹窗。这种多个对象共用同一个接口,又表现的形态不一样的现象,就叫做多态(Polymmorphism)。

Polymorphilsm is based on the greek words Poly (many) and morphism(forms), 接下来我们通过代码来演示什么是多态。

(1)通过统一函数接口实现多态

class Dog():def sound(self):print("汪汪汪...")class Cat():def sound(self):print("喵喵喵...")def make_sound(animal):animal.sound()dog = Dog()
cat = Cat()
make_sound(cat)  # 喵喵喵...
make_sound(dog)  # 汪汪汪...

(2)通过抽象类实现多态(最常用)

假如你开发一个文本编辑器,支持多种文档类型, 在用户通过你的编辑器打开文件之前,你也不知道准备要打开的是什么类型的文件,可能是pdf,也可能是word。
假如你为每个文件类型都写一个类,每个类都通过show()方法来调用打开对应的文档,为了确保每个类都必须实现show()方法,你可以写一个抽象类Document:

class Document():def __init__(self, name):self.name = namedef show(self):raise NotImplementedError("Subclass must implement abstract method")class Pdf(Document):def show(self):print("Show Pdf contents!")class Word(Document):def show(self):print("Show Word contents!")pdf = Pdf("1.pdf")
word = Word("2.doc")
obj = [pdf, word]
for o in obj:o.show()

四. 类与对象综合练习:校园管理系统

设计一个培训机构管理系统,有总部、分校,有学员、老师、员工,实现具体如下需求:
1.有多个课程,课程要有定价
2.有多个班级,班级跟课程有关联
3.有多个学生,学生报名班级,交这个班级对应的课程的费用
4.有多个老师,可以分布在不同校区,上不同班级的课
5.有多个员工,可以分布在不同校区在总部可以统计各校区的账户余额、员工人数、学员人数
6.学生可以转校、退学

随便瞎写了一个,可能略有出入,如有语法错误欢迎指正!

(此部分建议先自己动手写,如果写不出来再去看别人的代码)

class School():def __init__(self, name, address):self.name = nameself.address = addressself.branches = set()self.stuff = set()self.classes = set()def print_address(self):print("学校[%s]的地址是[%s]" % (self.name, self.address))def print_classes(self):print("学校[%s]目前所开班型有:" % self.name)for i in self.classes:print(i.class_name)def print_branches(self):print("学校[%s]目前的分校有:" % self.name)for i in self.branches:print(i.name)def print_stuff(self):print("学校[%s]目前的雇员有:" % self.name)for i in self.stuff:print(i.name)def pay_roll(self):self.count_teacher_num()self.count_stuff_num()money = 0for i in self.stuff:money += i.salaryfor i in self.classes:money += i.teacher.salaryprint("总校应发工资:%s" % money)for i in self.branches:for j in i.stuff:    money += j.salaryfor j in i.classes:money += j.teacher.salaryprint("总校分校合计应发工资:%s" % money)def count_teacher_num(self):teacher_num = len(self.classes)for i in self.branches:teacher_num += len(i.classes)print("[%s]有教师数量[%s]"%(self.name, teacher_num))def count_stuff_num(self):stuff_num = len(self.stuff)for i in self.branches:stuff_num += len(i.stuff)print("[%s]有职工数量[%s]"%(self.name, stuff_num))def count_stu_num(self):stu_num = 0for i in self.classes:stu_num += len(i.student)for i in self.branches:for j in i.classes:stu_num += len(j.student)print("[%s]有学生数量[%s]"%(self.name, stu_num))class BranchSchool():def __init__(self, name, address, headquater):self.name = nameself.address = addressself.headquater = headquaterheadquater.branches.add(self)self.classes = set()self.stuff = set()def print_address(self):print("学校[%s]的地址是[%s]" % (self.name, self.address))def print_classes(self):print("学校[%s]目前所开班型有:" % self.name)for i in self.classes:print(i.class_name)def print_stuff(self):print("学校[%s]目前的雇员有:" % self.name)for i in self.stuff:print(i.name)class Kecheng():def __init__(self, class_name, class_id, price, num_period, branchschool):self.class_name = class_nameself.class_id = class_idself.price = priceself.num_period = num_periodself.student = set()self.branchschool = branchschoolbranchschool.classes.add(self)self.teacher = Nonedef print_class_information(self):if self.teacher == None:print("本课程未指定授课教师,请调用教师类方法指定授课教师!")else:print("[%s]课程基本信息:" % self.class_name)print("课序号:[%s]" % self.class_id)print("开课校区:[%s]" % self.branchschool.name)print("授课教师:[%s]" % self.teacher.name)print("本课程学时:[%s]" % self.num_period)def print_student(self):self.print_class_information()print("-------本课程学生名单----------")for i in self.student:print(i.name,i.student_id)class Student():def __init__(self, name, student_id):self.name = nameself.student_id = student_idself.kecheng = Nonedef join_class(self, kecheng):self.kecheng = kechengself.Number_of_hours_attended = 0kecheng.student.add(self)print("学生[%s]选报了[%s]课程,课序号:[%s]" % (self.name, kecheng.class_name, kecheng.class_id))def print_information(self):print("学生[%s], 学号[%s]" % (self.name, self.student_id))if self.kecheng == None:print("该生没有选报课程!")else:print("该生所上课程[%s], 已上课时数[%s]" % (self.kecheng.class_name, self.Number_of_hours_attended))def shangke(self):self.Number_of_hours_attended += 1print("学生[%s]来上课了,已上课时数[%s]" % (self.name, self.Number_of_hours_attended))def tuixue(self):shengyuxueshi = self.kecheng.num_period - self.Number_of_hours_attendedmoney = shengyuxueshi * self.kecheng.priceprint("学生[%s]退课了,剩余课时数[%s],应退学费[%s]元" % (self.name, shengyuxueshi, money))self.kecheng.student.remove(self)class Teacher():def __init__(self, name, salary, teacher_id):self.name = nameself.salary = salaryself.kecheng = Noneself.teacher_id = teacher_iddef print_information(self):print("教师[%s], 教师工作证号[%s]" % (self.name, self.teacher_id))if self.kecheng == None:print("该老师不执教任何课程!")else:print("该老师所上课程[%s]" % (self.kecheng.name))def join_class(self, kecheng):self.kecheng = kechengkecheng.teacher = selfprint("教师[%s]成为课程[%s]的教师!课序号[%s]" % (self.name, kecheng.class_name, kecheng.class_id))class Stuff():def __init__(self, name, salary, stuff_id):self.name = nameself.salary = salaryself.stuff_id = stuff_idself.school = Nonedef print_information(self):print("雇员[%s], 工作证号[%s]" % (self.name, self.stuff_id))if self.school == None:print("该雇员还没有分配校区!")else:print("所在校区[%s]" % (self.school.name))def join_school(self, school):self.school = schoolprint("雇员[%s]加入[%s]校区" % (self.name, school.name))school.stuff.add(self)school1 = School("北京总校", "北京市海淀区清华大学")
branchschool1 = BranchSchool("北京分校", "北京站1号", school1)
school2 = School("上海总校", "上海市黄浦区南京东路1号")
branchschool2 = BranchSchool("上海分校", "虹桥火车站", school2)
branchschool3 = BranchSchool("上海第二分校", "松江大学城", school2)
student1 = Student("张三", 4312784)
student2 = Student("李四", 4371289)
student3 = Student("王五", 6578584)
student4 = Student("赵八", 3421554)
student5 = Student("曹一", 1254265)
student6 = Student("操日本", 9432713)
teacher1 = Teacher("赵老憨", 15000, 5478238)
teacher2 = Teacher("李老师", 18000, 8033427)
kecheng1 = Kecheng("数据结构", 548792, 500, 45, school1)
kecheng2 = Kecheng("计算机组成原理", 431287, 500, 45, branchschool1)
stuff1 = Stuff("王莹", 20000, 412379)
stuff2 = Stuff("赵鹏", 1000, 4127982)
stuff3 = Stuff("李麻瓜", 2000, 5189312)
school2.print_branches()
school1.print_classes()
student1.print_information()
branchschool1.print_classes()
student1.join_class(kecheng1)
student2.join_class(kecheng2)
student3.join_class(kecheng1)
teacher1.join_class(kecheng1)
teacher2.join_class(kecheng2)
kecheng1.print_student()
school1.count_stu_num()
student1.print_information()
student1.shangke()
student1.shangke()
student1.tuixue()
stuff1.join_school(school1)
stuff2.join_school(branchschool1)
stuff3.join_school(school2)
school1.print_stuff()
school1.pay_roll()

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

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

相关文章

新手淘宝开店如何引流

对于新手淘宝卖家来说&#xff0c;引流是开店过程中最为关键的一环。如何吸引潜在客户进入店铺&#xff0c;提高商品的曝光率和销量&#xff0c;是每个新手卖家都面临的挑战。本文将为你提供新手淘宝开店的引流攻略&#xff0c;帮助你从零开始掌握实用的引流技巧。 一、优化店…

C++的类型转换

1.C语言中的类型转换 在C语言中&#xff0c;如果赋值运算符左右两侧类型不同&#xff0c;或者形参与实参类型不匹配&#xff0c;或者返回值类型与接收返回值类型不一致时&#xff0c;就需要发生类型转化&#xff0c;C语言中总共有两种形式的类型转换&#xff1a;隐式类型转换和…

【机器人最短路径规划问题(栅格地图)】基于模拟退火算法求解

代码获取方式&#xff1a;QQ&#xff1a;491052175 或者 私聊博主获取 基于模拟退火算法求解机器人最短路径规划问题&#xff08;栅格地图&#xff09;的仿真结果 仿真结果&#xff1a; 初始解的路径规划图 收敛曲线&#xff1a; 模拟退火算法求解的路径规划图 结论&#xff…

Ubuntu20安装zabbix-agent2,对接zabbix 6.4

在Ubuntu 20.04 LTS上安装Zabbix Agent 2并与Zabbix Server 6.4对接&#xff0c;请按照以下步骤操作&#xff1a; 更新系统&#xff1a; sudo apt update sudo apt upgrade 添加Zabbix官方仓库&#xff1a; 首先&#xff0c;需要将Zabbix的官方存储库添加到你的系统中以获取Za…

C#面:常用的 异常类 有哪些

异常类是用于处理程序运行时出现的错误或异常情况的类。 C# 提供了一些内置的异常类&#xff0c;常用的包括&#xff1a; System.Exception&#xff1a;所有异常类的基类&#xff0c;可以用于捕获所有类型的异常。System.SystemException&#xff1a;表示系统级别的异常&…

【了解SpringCloud Gateway微服务网关】

曾梦想执剑走天涯&#xff0c;我是程序猿【AK】 目录 简述概要知识图谱什么是SpringCloudGateway功能特征应用场景核心概念配置文件工作原理路由谓词工厂&#xff08;内置的&#xff09;[After 路由谓词工厂](https://docs.spring.io/spring-cloud-gateway/docs/current/refere…

Mysql运维篇(七) 部署MHA--完结

一路走来&#xff0c;所有遇到的人&#xff0c;帮助过我的、伤害过我的都是朋友&#xff0c;没有一个是敌人。如有侵权&#xff0c;请留言&#xff0c;我及时删除&#xff01; 一、MHA软件构成 Manager工具包主要包括以下几个工具&#xff1a; masterha_manger 启…

【C++】多态深入分析

目录 一&#xff0c;多态的原理 1&#xff0c;虚函数表与虚函数表指针 2&#xff0c;原理调用 3&#xff0c;动态绑定与静态绑定 二&#xff0c;抽象类 三&#xff0c;单继承和多继承关系的虚函数表 1&#xff0c;单继承中的虚函数表 2&#xff0c;多继承中的虚函数表 …

“编码迷宫中的探险者:探索程序员职业赛道的无限可能“

在这个信息技术飞速发展的时代&#xff0c;程序员的职业赛道就像是一座错综复杂的迷宫&#xff0c;它既充满了挑战&#xff0c;又蕴藏着无限的机遇。这座迷宫中&#xff0c;有前端的美丽花园&#xff0c;后端的黑暗洞穴&#xff0c;还有数据科学的神秘密室。每一条路径都有其独…

内网搭建mysql8.0并搭建主从复制详细教程!!!

一、安装mysql 1.1 mysql下载链接&#xff1a; https://downloads.mysql.com/archives/community/ 1.2 解压包并创建相应的数据目录 tar -xvf mysql-8.2.0-linux-glibc2.28-x86_64.tar.xz -C /usr/local cd /usr/local/ mv mysql-8.2.0-linux-glibc2.28-x86_64/ mysql mkdir…

Python绘图-9饼图(上)

饼图&#xff08;Pie Chart&#xff09;是一种用于表示数据分类和相对大小的可视化图形。在饼图中&#xff0c;整个圆形代表数据的总和&#xff0c;而圆形内的各个扇形则代表不同的分类或类别&#xff0c;扇形的面积大小表示该类别在整体中所占的比例。饼图通常用于展示数据的分…

FW, IPS, IDS

文章目录 FW (Firewall, 防火墙)IPS (Intrusion Prevention System, 入侵防御系统)IDS (Intrusion Detection System, 入侵检测系统)IDS vs. FWIPS FW (Firewall, 防火墙) 产品定位&#xff1a; 防火墙的主要作用是进行网络访问控制。它充当网络的门卫&#xff0c;控制进入和离…

《人间值得》读书笔记

人的一生说短不短&#xff0c;说长不长。芸芸众生&#xff0c;为了生活努力的挣扎&#xff0c;太少的人能衣食无忧&#xff0c;所以我们每天为了碎银几两&#xff0c;为了生活奔波。 《人间值得》的主人公是一个90岁的老奶奶&#xff0c;她的生活经历很丰富&#xff0c;她的人…

ObjectProvider学习

简介 ObjectProvider 是 Spring Framework 5.0 之后引入的一个新接口&#xff0c;它提供了一个灵活的方式来访问由 Spring 容器管理的 Bean。ObjectProvider 提供了一种更加类型安全的方式来查找和注入依赖项&#xff0c;同时还支持 Null 值的处理以及延迟初始化。 ObjectProv…

Window部署Jaeger

参考&#xff1a;windows安装使用jaeger链路追踪_windows安装jaeger-CSDN博客 下载&#xff1a;Releases jaegertracing/jaeger GitHub Jaeger – Download Jaeger 目录 1、安装nssm 2、安装运行 elasticsearch 3、安装运行 3.1部署JaegerAgent 3.2部署JaegerCollec…

【全志D1-H 哪吒开发板】Debian系统安装调教和点灯指南

全志D1-H开发板【哪吒】使用Deabian系统入门 特别说明&#xff1a; 因为涉及到操作较多&#xff0c;博文可能会导致格式丢失 其中内容&#xff0c;会根据后续使用做优化调整 目录&#xff1a; 参考资料固件烧录启动调教点灯问题 〇、参考资料 官方资料 开发板-D1开发板【…

C++:函数模板整理

函数模板: 找到函数相同的实现思路&#xff0c;区别于函数的参数类型。 使用函数模板使得函数可容纳不同类型的参数实现函数功能&#xff0c;而不是当类型不同时便编译大量类型不同的函数&#xff0c;产生大量重复代码和内存占用 函数模板格式&#xff1a; template<typ…

[Vulnhub]靶场 Red

kali:192.168.56.104 主机发现 arp-scan -l # arp-scan -l Interface: eth0, type: EN10MB, MAC: 00:0c:29:d2:e0:49, IPv4: 192.168.56.104 Starting arp-scan 1.10.0 with 256 hosts (https://github.com/royhills/arp-scan) 192.168.56.1 …

ARM64汇编02 - 寄存器与指令基本格式

最近的文章可能会有较多修改&#xff0c;请关注博客哦 异常级别 ARMv8处理器支持4种异常等级&#xff08;Exception Level&#xff0c;EL&#xff09;。 EL0 为非特权模式&#xff0c;用于运行应用程序&#xff0c;其他资源访问受限&#xff0c;权限不够。 EL1 为特权模式&…

【王道操作系统】ch1计算机系统概述-06虚拟机

文章目录 【王道操作系统】ch1计算机系统概述-06虚拟机01传统计算机02虚拟机的基本概念&#xff08;1&#xff09;第一类虚拟机管理程序&#xff08;2&#xff09; 第二类虚拟机管理程序&#xff08;3&#xff09; 两类虚拟机管理程序的对比 【王道操作系统】ch1计算机系统概述…