python之面向对象性封装,多态,以及鸭子类型

2022-11-20,,,,

默认类型

class A:

    class_name = 'python23期'

    def __init__(self, name, age):

        self.name = name
self.age =age a1 = A('李业', 21) # 实例化一个a1的对象
print(a1.name) a2 = A('李东宇', 24) # 实例化一个a2的对象
print(a2.age)

封装

把很多数据封装到⼀个对象中. 把固定功能的代码封装到⼀个代码块, 函数, 对象, 打包成模块. 这都属于封装的思想. 具体的情况具体分析. 比如. 你写了⼀个很⽜B的函数. 那这个也可以被称为封装. 在⾯向对象思想中. 是把⼀些看似⽆关紧要的内容组合到⼀起统⼀进⾏存储和使⽤. 这就是封装.

多态

一个事务有多种形态 比如水

什么是多态: 在python中默认支持多态,就是a可以定义多种数据类型,不用规定其所属的数据类型,可以是字符串,列表,元组等任意数据类型

python中定义变量不用规定变量的类型
a = 'alex'
a = [1, 2, 3]
a = (22, 33) Java
int a = 12 # a必须是整型
String b = 'abc'# b必须是一个字符串类型的

鸭子类型

python中你看起来像鸭子,那么你就是鸭子

就是相互独立的两个类,本身是没有什么关系的,然后它们内部有共同的名字func,这种就是鸭子

格式

class A:

    def login(self):
pass def register(self):
pass class B: def login(self):
pass def register(self):
pass
# A,B两个类,没有任何关系,独立两个,但是里面的功能相似,所以python一般会将类似于A,B两个类
# 里面的相似的功能让其命名相同.
# 1. A,B虽然无关系,但是很默契的制定了一个规范.让你使用起来更方便.

super

格式

class A:
def f1(self):
print('in A f1') def f2(self):
print('in A f2') class Foo(A):
def f1(self):
# super().f2()
super(Foo, self).f2()
print('in A Foo') obj = Foo()
obj.f1()
# 执行结果
in A f2
in A Foo class A:
def f1(self):
print('in A') class Foo(A):
def f1(self):
super(Foo,self).f1()
print('in Foo') # 2 class Bar(A):
def f1(self):
print('in Bar') # 1 class Info(Foo,Bar): def f1(self):
super(Info,self).f1()
print('in Info f1') # 3 obj = Info()
print(Info.mro()) # [Info, Foo, Bar, A]
obj.f1()
# 执行结果
[<class '__main__.Info'>, <class '__main__.Foo'>, <class '__main__.Bar'>, <class '__main__.A'>, <class 'object'>]
in Bar
in Foo
in Info f1

super() 严格意义并不是执行父类的方法.

单继承: super() 肯定是执行父类的方法.

多继承: super(S,self) 严格按照self从属于的类的mro的执行顺序,执行 S类的下一位.***

类的约束

# 版本二 统一接口
class Wecht: def pay(self,money):
print(f'利用微信支付了{money}') class Alipay: def pay(self, money):
print(f'利用支付宝支付了{money}') def pay(obj, money): # 定义一个统一化的设计
obj.pay(money) obj1 = Wecht()
obj2 = Alipay() pay(obj1,200)
pay(obj2,300) # 输出结果
利用微信支付了200
利用支付宝支付了300 # 版本三 野路子写法 # 版本四 按照之前的代码逻辑进行改变 发现问题:
以上代码没有约束,原因就是想怎么写就怎么写,都能实现当时的功能(个人理解)
在上面的情况下(在一些重要的逻辑,与用户数据相关等核心部分),我们要建立一种约束,避免发生此类错误

类的约束有两种解决方式

    在父类建立一种约束
    利用抽象类(指定的一种规范)的概念

第一种解决方式(建议使用方式)

在父类定义一个pay方法,主动抛出异常,如果子类没有定义pay方法,并且

再建立一个父类pay的方法,约定俗称定义一种规范,子类要定义pay方法,但是没有强制性,还是可以执行
raise 主动报错提醒
class Payment:
def pay(self,money):
raise Exception('必须得定义这个类') class Wechat(Payment): def pay(self,money):
print(f'利用微信支付了{money}') class Alipay(Payment): def pay(self, money):
print(f'利用支付宝支付了{money}') class QQpay(Payment): def fuqian(self, money):
print(f'利用QQ支付了{money}') def pay(obj, money):
obj.pay(money) obj1 = Wechat()
obj2 = Alipay() pay(obj1,200)
pay(obj2,300) obj3 = QQpay()
pay(obj3,599) # 输出结果
Traceback (most recent call last):
File "/Users/wuqiang/work/PycharmProjects/python23/day24/day24.py", line 50, in <module>
pay(obj3,599)
File "/Users/wuqiang/work/PycharmProjects/python23/day24/day24.py", line 41, in pay
obj.pay(money)
File "/Users/wuqiang/work/PycharmProjects/python23/day24/day24.py", line 22, in pay
raise Exception('必须得定义这个类')
Exception: 必须得定义这个类

第二种解决方式

利用抽象类概念, 基类如先设置,子类如果没有定义pay方法,在实例化对象的时候就会抛出错误

from abc import ABCMeta,abstractmethod
class Payment(metaclass=ABCMeta):
@abstractmethod
def pay(self, money):
pass class Wecht(Payment): def pay(self,money):
print(f'利用微信支付了{money}') class Alipay(Payment): def pay(self, money):
print(f'利用支付宝支付了{money}') class QQpay(Payment): def fuqian(self, money):
print(f'利用QQ支付了{money}') obj3 = QQpay()
# 输出结果
obj3 = QQpay()
TypeError: Can't instantiate abstract class QQpay with abstract methods pay

python之面向对象性封装,多态,以及鸭子类型的相关教程结束。

《python之面向对象性封装,多态,以及鸭子类型.doc》

下载本文的Word格式文档,以方便收藏与打印。