Python——彻底明白Super() 该如何使用

什么是Super()

如果把类比作表演者的话,Super()则是按照清单(MRO)让下一位表演者"继续"他的表演

没听明白没关系,我们通过实际的例子来逐步拆解Super()的运行逻辑,到时候我们就能明白这句话的含义

为什么要用Super()

和拆解Property一样,我们在拆解Super()前也要知道我们为什么要用它

我们同样通过实际的代码示例来理解

python 复制代码
class BasicOrder:
    def __init__(self, price):
        self.price = price
    
    def get_total(self):
        return self.price
        # 返回基础价格
class MemberOrder(BasicOrder):
    def get_total(self):
        return self.price * 0.9
        # 返回会员折后价
        
class PromotionOrder(BasicOrder):
    def get_total(self):
        if self.price > 200:
            return self.price - 20
            # 满减条件
        else:
            return self.price
        
a = BasicOrder(300)
print(a.get_total())
# 原价
b = MemberOrder(300)
print(b.get_total())
# 会员折后价
c = PromotionOrder(300)
print(c.get_total())
# 满减价格

我们定义了两种折扣机制,一种是会员价,一种是满减。我们可以根据不同的折扣方式,算出打折后的价格

但是,有一种情况我们并没有考虑,那就是两种折扣同时触发的情况。如果我们规定,当两个折扣方式同时满足时,先计算会员折后价,再考虑满减,我们该如何修改这段代码呢?

很简单,我们再定义一个类,就像这样

python 复制代码
class BasicOrder:
    def __init__(self, price):
        self.price = price

    def get_total(self):
        return self.price
        # 返回基础价格


class MemberOrder(BasicOrder):
    def get_total(self):
        return self.price * 0.9
        # 返回会员折后价


class PromotionOrder(BasicOrder):
    def get_total(self):
        if self.price > 200:
            return self.price - 20
        else:
            return self.price
        # 触发满减

class FinalOrder(MemberOrder):
    def get_total(self):
        member_price = MemberOrder.get_total(self)
        if member_price > 200:
            return member_price - 20
        else:
            return member_price
        
a = BasicOrder(300)
print(a.get_total())
# 原价
b = MemberOrder(300)
print(b.get_total())
# 会员折后价
c = PromotionOrder(300)
print(c.get_total())
# 满减价格
d = FinalOrder(300)
print(d.get_total())
# 双重折扣后的价格

如果要更换折扣顺序,先计算满减,再计算会员价,只要对FinalOrder进行修改就好

python 复制代码
class FinalOrder(PromotionOrder):
    def get_total(self):
        count_price = PromotionOrder.get_total(self)
        return count_price * 0.9

我们这种修改,在类的数量也就是折扣方式比较少时没什么问题。但如果类的数量比较多,我们就要想办法,如何不用进行这么复杂的修改。这就是我们使用super()的原因

Super()的作用

我们用super()对上面的代码进行优化

python 复制代码
class BasicOrder:
    def __init__(self, price):
        self.price = price

    def get_total(self):
        return self.price
        # 返回基础价格


class MemberOrder(BasicOrder):
    def get_total(self):
        return super().get_total() * 0.9
        # 返回会员折后价


class PromotionOrder(BasicOrder):
    def get_total(self):
        if super().self.price > 200:
            return super().get_total() - 20
        else:
            return super().get_total()
        # 触发满减

class FinalOrder(MemberOrder, PromotionOrder):
    def get_total(self):
        return super().get_total()
        
a = BasicOrder(300)
print(a.get_total())
# 原价
b = MemberOrder(300)
print(b.get_total())
# 会员折后价
c = PromotionOrder(300)
print(c.get_total())
# 满减价格
d = FinalOrder(300)
print(d.get_total())
# 双重折扣后的价格
print(FinalOrder.mro())
# 打印Mro清单

我们要注意最后的Mro清单,它的输出结果是[<class '__main__.FinalOrder'>, <class '__main__.MemberOrder'>, <class '__main__.PromotionOrder'>, <class '__main__.BasicOrder'>, <class 'object'>]。这也就是当我们计算FinalOrder中的最后价格时,程序的执行顺序

python按照这个顺序,先从FinalOrder-MemberOrder-PromotionOrder-BasicOrder这个链条进行递进寻找,然后再从反方向BasicOrder-PromotionOrder-MemberOrder-FinalOrder进行回归计算

上面的代码就是按照这个顺序,先把原价BasicOrderPromotionOrder计算满减,再将满减后的值给MemberOrder计算会员折扣价,最后把值给FinalOrder

return super().get_total()其实就可以理解为请MRO链条上的下一位先算出结果给我

如果我们需要改变折扣计算的方式,我们也无须对FinalOrder中的内容进行修改,只需要修改继承顺序,也就是对这个"链条"进行调整就好

当我们修改某些类名时,我们也不用像之前的"硬编码"那样逐个修改,只要确保继承的类名正确,super()会自己按照MRO这个链条进行寻找

python 复制代码
class FinalOrder(PromotionOrder, MemberOrder):
    def get_total(self):
        return super().get_total()

结语

如果文章中有什么错误或者可以优化的地方,欢迎大家指正,感谢!

相关推荐
小猪咪piggy3 小时前
【Python】(4) 列表和元组
开发语言·python
墨理学AI4 小时前
一文学会一点python数据分析-小白原地进阶(mysql 安装 - mysql - python 数据分析 - 学习阶段梳理)
python·mysql·数据分析
数研小生4 小时前
亚马逊商品列表API详解
前端·数据库·python·pandas
独好紫罗兰4 小时前
对python的再认识-基于数据结构进行-a005-元组-CRUD
开发语言·数据结构·python
jianghua0014 小时前
Python中的简单爬虫
爬虫·python·信息可视化
喵手4 小时前
Python爬虫实战:针对Python官网,精准提取出每一个历史版本的版本号、发布日期以及对应的文档/详情页链接等信息,并最终清洗为标准化的CSV文件!
爬虫·python·爬虫实战·零基础python爬虫教学·python官方数据采集·采集历史版本版本号等信息·导出csv文件
databook4 小时前
像搭积木一样思考:数据科学中的“自下而上”之道
python·数据挖掘·数据分析
luoluoal4 小时前
基于python的医疗问句中的实体识别算法的研究(源码+文档)
python·mysql·django·毕业设计·源码
啊阿狸不会拉杆5 小时前
《机器学习导论》第 9 章-决策树
人工智能·python·算法·决策树·机器学习·数据挖掘·剪枝