python中的类与对象(3)

news2024/11/15 11:29:29

目录

一. 类的多继承

二. 类的封装

三. 类的多态

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


一. 类的多继承

在(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):
    pass

class 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):
    pass

class E(C, C2):
    pass

class F(D, E):
    pass

f1 = 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 = name
        self.sex = sex
        self.life_val = 100

a = 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 = sex
        self.__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 = name
        self.sex = sex
        self.__life_val = 100  
    def get_life_val(self):
        print("生命值还有:", self.__life_val)
    def got_attack(self):
        self.__life_val -= 20
        print("[%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 = name
        self.sex = sex
        self.__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 = sex
        self.__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 = name
    def 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 = name
        self.address = address
        self.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 = 0
        for i in self.stuff:
            money += i.salary
        for i in self.classes:
            money += i.teacher.salary
        print("总校应发工资:%s" % money)
        for i in self.branches:
            for j in i.stuff:    
                money += j.salary
            for j in i.classes:
                money += j.teacher.salary
        print("总校分校合计应发工资:%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 = 0
        for 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 = name
        self.address = address
        self.headquater = headquater
        headquater.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_name
        self.class_id = class_id
        self.price = price
        self.num_period = num_period
        self.student = set()
        self.branchschool = branchschool
        branchschool.classes.add(self)
        self.teacher = None
    def 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 = name
        self.student_id = student_id
        self.kecheng = None
    def join_class(self, kecheng):
        self.kecheng = kecheng
        self.Number_of_hours_attended = 0
        kecheng.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 += 1
        print("学生[%s]来上课了,已上课时数[%s]" % (self.name, self.Number_of_hours_attended))
    def tuixue(self):
        shengyuxueshi = self.kecheng.num_period - self.Number_of_hours_attended
        money = shengyuxueshi * self.kecheng.price
        print("学生[%s]退课了,剩余课时数[%s],应退学费[%s]元" % (self.name, shengyuxueshi, money))
        self.kecheng.student.remove(self)

class Teacher():
    def __init__(self, name, salary, teacher_id):
        self.name = name
        self.salary = salary
        self.kecheng = None
        self.teacher_id = teacher_id
    def 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 = kecheng
        kecheng.teacher = self
        print("教师[%s]成为课程[%s]的教师!课序号[%s]" % (self.name, kecheng.class_name, kecheng.class_id))

class Stuff():
    def __init__(self, name, salary, stuff_id):
        self.name = name
        self.salary = salary
        self.stuff_id = stuff_id
        self.school = None
    def 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 = school
        print("雇员[%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.coloradmin.cn/o/1486256.html

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

相关文章

Latex中如何调整算法伪代码\begin{algorithm}的字体大小?

在LaTeX中&#xff0c;要调整algorithm环境&#xff08;通常与algorithmic、algorithmicx、algorithm2e等包一起使用来编写伪代码&#xff09;中的字体大小&#xff0c;你可以使用\small、\footnotesize、\tiny等命令来减小字体大小&#xff0c;或者使用\large、\Large、\LARGE…

简单实现文字滚动效果-CSS版本

先看看效果 话不多说直接上代码 <template><div class"main"><div class"scroll-region"><div class"swiper-scroll-content"><span class"list-btn" v-for"(item, index) in overviewList" :…

【重温设计模式】命令模式及其Java示例

命令模式的介绍 在我们的编程世界中&#xff0c;设计模式如同一盏指路明灯&#xff0c;它们是前人智慧的结晶&#xff0c;为我们照亮了解决复杂问题的路径。 其中&#xff0c;命令模式是一种行为设计模式&#xff0c;其主要的目标是将操作封装到对象中。这种模式的一个显著特点…

基于springboot+vue的图书电子商务网站

博主主页&#xff1a;猫头鹰源码 博主简介&#xff1a;Java领域优质创作者、CSDN博客专家、阿里云专家博主、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战&#xff0c;欢迎高校老师\讲师\同行交流合作 ​主要内容&#xff1a;毕业设计(Javaweb项目|小程序|Pyt…

数据修改

Oracle 目录 数据修改 将员工编号的 7369 的员工工资修改为 810&#xff0c;佣金改为 100 将工资最低的员工工资修改为公司的平均工资 将所有在 1981 年雇佣的员工的雇佣日期修改为今天&#xff0c;工资增长 20% 数据的更新操作 Oracle从入门到总裁:https://blog.csdn.n…

学习人工智能的方法及方向!

目录 一、第一部分&#xff1a;了解人工智能 二、人工智能学习路线图 三、职业规划 四、未来展望 五、总结 在这个信息爆炸的时代&#xff0c;想要系统性地学习人工智能&#xff08;AI&#xff09;并找到对应方向的工作&#xff0c;你需要一个明确的学习路径和职业规划。本…

【latex】\IEEEpubid版权声明与正文内容重叠

问题描述 撰写IEEE Trans论文时&#xff0c;出现版权声明文字\IEEEpubid与正文内容重叠的问题&#xff1a; 原因分析&#xff1a; 在使用模板时&#xff0c;不小心将以下命令删除了&#xff1a; \IEEEpubidadjcol 解决方案&#xff1a; 在需要换页的位置附近添加以上命令&…

投影和定义投影的区别

Arcmap中关于投影的工具有四个&#xff0c;分别是定义投影、投影、投影栅格、批量投影。这四个工具既有相同之处也有不同之处&#xff0c;下面我将一一介绍。 ①定义投影&#xff1a;Arcmap中关于定义投影工具是这样描述的&#xff1a;“所有地理数据集均具有一个用于显示、测…

【LeetCode题解】2859. 计算 K 置位下标对应元素的和+938. 二叉搜索树的范围和+1028. 从先序遍历还原二叉树(三种方法:栈+递归+集合)

文章目录 [2859. 计算 K 置位下标对应元素的和](https://leetcode.cn/problems/sum-of-values-at-indices-with-k-set-bits/)思路&#xff1a; [938. 二叉搜索树的范围和](https://leetcode.cn/problems/range-sum-of-bst/)思路&#xff1a;写法一&#xff1a;在中间累加写法二…

从零开始学习Netty - 学习笔记 -Netty入门【半包,黏包】

Netty进阶 1.黏包半包 1.1.黏包 服务端代码 public class HelloWorldServer {private static final Logger logger LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());public static void main(String[] args) {NioEventLoopGroup bossGroup new NioEventL…

Linux多线程控制:深入理解与应用(万字详解!)

&#x1f3ac;慕斯主页&#xff1a;修仙—别有洞天 ♈️今日夜电波&#xff1a;どうして (feat. 野田愛実) 0:44━━━━━━️&#x1f49f;──────── 3:01 &#x1f504; ◀️ ⏸ ▶️ …

基于redis实现【最热搜索】和【最近搜索】功能

目录 一、前言二、分析问题三、针对两个问题&#xff0c;使用redis怎么解决问题&#xff1f;1、字符串String2、列表List3、字典Hash4、集合Set5、有序集合ZSet6、需要解决的五大问题 四、编写代码1.pom依赖2.application.yml配置3.Product商品实体4.用户最近搜索信息5.redis辅…

C-V2X系列:C-V2X芯片及模组整理总结

C-V2X、车路协同、车联网、智能网联车学习 C-V2X芯片及模组整理总结

Typora旧版链接(Win+Mac+Linux版)

记得点赞本文&#xff01;&#xff01;&#xff01; 链接&#xff1a;https://pan.baidu.com/s/1IckUvQUBzQkfHNHXla0zkA?pwd8888 提取码&#xff1a;8888 –来自百度网盘超级会员V7的分享

2.模拟问题——4.日期问题

日期问题难度并不大&#xff0c;但是代码量非常大&#xff0c;需要较高的熟练度&#xff0c;因此需要着重练习&#xff0c;主要涉及数组和循环两个方面的知识点&#xff0c;需要熟练的测试代码。 两个经典题型 闰年 闰年满足以下两个条件的任意一个 能够被400整除不能够被1…

Golang Vs Java:为您的下一个项目选择正确的工具

Java 首次出现在 1995 年&#xff0c;由 James Gosling 和 Sun Microsystems 的其他人开发的一种新编程语言。从那时起&#xff0c;Java 已成为世界上最受欢迎和广泛使用的编程语言之一。Java 的主要特点包括其面向对象的设计、健壮性、平台独立性、自动内存管理以及广泛的内置…

JavaSec 基础之 JNDI 注入

文章目录 JNDI简介JNDI 支持的服务协议JNDI 注入JNDI 复现修复 JNDI 简介 JNDI(Java Naming and Directory Interface)是一个应用程序设计的 API&#xff0c;一种标准的 Java 命名系统接口。JNDI 提供统一的客户端 API&#xff0c;通过不同的访问提供者接口JNDI服务供应接口(…

武器大师——操作符详解(下)

目录 六、单目操作符 七、逗号表达式 八、下标引用以及函数调用 8.1.下标引用 8.2.函数调用 九、结构体 9.1.结构体 9.1.1结构的声明 9.1.2结构体的定义和初始化 9.2.结构成员访问操作符 9.2.1直接访问 9.2.2间接访问 十、操作符的属性 10.1.优先性 10.2.结合性 …

Ubuntu20.04使用XRDP安装原生远程桌面

Ubuntu20.04使用XRDP安装原生远程桌面 1.安装gnome桌面 # 如果没有更新过源缓存&#xff0c;先更新一下 sudo apt update# 安装gnome桌面 # 可选参数 --no-install-recommends&#xff0c;不安装推荐组件&#xff0c;减少安装时间和空间占用 sudo apt install ubuntu-desktop…

2.2_5 调度算法

文章目录 2.2_5 调度算法一、适用于早期的批处理系统&#xff08;一&#xff09;先来先服务&#xff08;FCFS&#xff0c;First Come First Serve&#xff09;&#xff08;二&#xff09;短作业优先&#xff08;SJF&#xff0c;Shortest Job First&#xff09;&#xff08;三&a…