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

默认类型

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

# 版本三 野路子写法

# 版本四 按照之前的代码逻辑进行改变

发现问题:
以上代码没有约束,原因就是想怎么写就怎么写,都能实现当时的功能(个人理解)
在上面的情况下(在一些重要的逻辑,与用户数据相关等核心部分),我们要建立一种约束,避免发生此类错误

类的约束有两种解决方式

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

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

在父类定义一个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
posted @ 2019-07-22 18:18  爱咋闹  阅读(283)  评论(0编辑  收藏  举报