• 封装和继承


    1. 封装

    ​ 封装是面向对象三大特性最核心的一个特性

    ​ 封装指的就是把数据与功能都整合到一起,封装<->整合。

    ​ 针对封装到对象或者类中的属性,严格控制对它们的访问{隐藏与开放接口}

    2. 隐藏属性

    Python的Class机制采用双下划线开头的方式将属性隐藏起来(设置成私有的),但其实这仅仅只是一种变形操作,类中所有双下滑线开头的属性都会在类定义阶段、检测语法时自动变成“_类名__属性名”的形式:

    class Foo:
        __N=0 # 变形为_Foo__N
    
        def __init__(self): # 定义函数时,会检测函数语法,所以__开头的属性也会变形
            self.__x=10 # 变形为self._Foo__x
    
        def __f1(self): # 变形为_Foo__f1
            print('__f1 run')
    
        def f2(self):  # 定义函数时,会检测函数语法,所以__开头的属性也会变形
            self.__f1() #变形为self._Foo__f1()
    
    print(Foo.__N) # 报错AttributeError:类Foo没有属性__N
    
    obj = Foo()
    print(obbj.__x) # 报错AttributeError:对象obj没有属性__x
    

    2.1 隐藏时注意的问题:

    I:在类外部无法直接访问双下滑线开头的属性,但知道了类名和属性名就可以拼出名字:类名__属性,然后就可以访问了,如Foo._A__N,
    所以说这种操作并没有严格意义上地限制外部访问,仅仅只是一种语法意义上的变形。

    class Foo:
        __x = 1  # _Foo__x
    
        def __f1(self):  # _Foo__f1
            print('from test')
    
    
    print(Foo.__dict__)
    print(Foo._Foo__x)
    print(Foo._Foo__f1)
    {'__module__': '__main__', '_Foo__x': 1, '_Foo__f1': <function Foo.__f1 at 0x0000017D78BFB310>, '__dict__': <attribute '__dict__' of 'Foo' objects>, '__weakref__': <attribute '__weakref__' of 'Foo' objects>, '__doc__': None}
    1
    <function Foo.__f1 at 0x0000017D78BFB310>
    

    II:这种隐藏对外不对内,因为__开头的属性会在检查类体代码语法时统一发生变形

    class Foo:
        __x = 1  # _Foo__x = 1
    
        def __f1(self):  # _Foo__f1
            print('from test')
    
        def f2(self):
            print(self.__x) # print(self._Foo__x)
            print(self.__f1) # print(self._Foo__f1)
    
    print(Foo.__x)#AttributeError: type object 'Foo' has no attribute '__x'
    print(Foo.__f1)#AttributeError: type object 'Foo' has no attribute '__f1'
    obj=Foo()
    obj.f2()
    

    III: 这种变形操作只在检查类体语法的时候发生一次,之后定义的__开头的属性都不会变形

    class Foo:
        __x = 1  # _Foo__x = 1
    
        def __f1(self):  # _Foo__f1
            print('from test')
    
        def f2(self):
            print(self.__x) # print(self._Foo__x)
            print(self.__f1) # print(self._Foo__f1)
    
    Foo.__y=3
    print(Foo.__dict__)
    print(Foo.__y)
    
    {'__module__': '__main__', '_Foo__x': 1, '_Foo__f1': <function Foo.__f1 at 0x0000019CBFD9B310>, 'f2': <function Foo.f2 at 0x0000019CBFD9B430>, '__dict__': <attribute '__dict__' of 'Foo' objects>, '__weakref__': <attribute '__weakref__' of 'Foo' objects>, '__doc__': None, '__y': 3}
    3
    

    2.2 隐藏的目的

    ​ I、隐藏数据属性"将数据隐藏起来就限制了类外部对数据的直接操作,然后类内应该提供相应的接口来允许类外部间接地操作数据,接口之上可以附加额外的逻辑来对数据的操作进行严格地控制:

    ​ II、隐藏函数/方法属性:目的的是为了隔离复杂度

    class People:
        def __init__(self, name):
            self.__name = name
    
        def get_name(self):
            # 通过该接口就可以间接地访问到名字属性
            print(self.__name)
    
        def set_name(self,val):
            if type(val) is not str:
                print('必须传字符串类型')
                return
            self.__name=val
    

    3. 开放接口

    定义属性就是为了使用的,所以隐藏不是目的而是为了特殊使用

    3.1 隐藏数据属性

    将数据隐藏起来就限制了类外部对数据的直接操作,然后类内应该提供相应的接口来允许类外部间接地操作数据,接口之上可以附加额外的逻辑来对数据的操作进行严格地控制

    >>> class Teacher:
    ...     def __init__(self,name,age): #将名字和年纪都隐藏起来
    ...         self.__name=name
    ...         self.__age=age
    ...     def tell_info(self): #对外提供访问老师信息的接口
    ...         print('姓名:%s,年龄:%s' %(self.__name,self.__age))
    ...     def set_info(self,name,age): #对外提供设置老师信息的接口,并附加类型检查的逻辑
    ...         if not isinstance(name,str):
    ...             raise TypeError('姓名必须是字符串类型')
    ...         if not isinstance(age,int):
    ...             raise TypeError('年龄必须是整型')
    ...         self.__name=name
    ...         self.__age=age
    ... 
    >>>
    >>> t=Teacher('lili',18)
    >>> t.set_info(‘LiLi','19') # 年龄不为整型,抛出异常
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
      File "<stdin>", line 11, in set_info
    TypeError: 年龄必须是整型
    >>> t.set_info('LiLi',19) # 名字为字符串类型,年龄为整形,可以正常设置
    >>> t.tell_info() # 查看老师的信息
    姓名:LiLi,年龄:19
    

    3.2 隐藏函数属性

    目的的是为了隔离复杂度,例如ATM程序的取款功能,该功能有很多其他功能组成,比如插卡、身份认证、输入金额、打印小票、取钱等,而对使用者来说,只需要开发取款这个功能接口即可,其余功能我们都可以隐藏起来

    >>> class ATM:
    ...     def __card(self): #插卡
    ...         print('插卡')
    ...     def __auth(self): #身份认证
    ...         print('用户认证')
    ...     def __input(self): #输入金额
    ...         print('输入取款金额')
    ...     def __print_bill(self): #打印小票
    ...         print('打印账单')
    ...     def __take_money(self): #取钱
    ...         print('取款')
    ...     def withdraw(self): #取款功能
    ...         self.__card()
    ...         self.__auth()
    ...         self.__input()
    ...         self.__print_bill()
    ...         self.__take_money()
    ...
    >>> obj=ATM()
    >>> obj.withdraw()
    

    总结隐藏属性与开放接口,本质就是为了明确地区分内外,类内部可以修改封装内的东西而不影响外部调用者的代码;而类外部只需拿到一个接口,只要接口名、参数不变,则无论设计者如何改变内部实现代码,使用者均无需改变代码。这就提供一个良好的合作基础,只要接口这个基础约定不变,则代码的修改不足为虑。

    4 property装饰器

    4.1 装饰器定义

    装饰器是在不修改装饰对象的源代码以及调用方式的前提下为被装饰对象添加新的功能的可调用对象

    4.2 什么是property

    property是一个用类实现的装饰器,是用来绑定给对象的方法伪造成一个数据属性

    print(property)
    <class 'property'>
    
    

    4.3 应用案例

    # 案例一:
    """
    成人的BMI数值:
    过轻:低于18.5
    正常:18.5-23.9
    过重:24-27
    肥胖:28-32
    非常肥胖, 高于32
      体质指数(BMI)=体重(kg)÷身高^2(m)
      EX:70kg÷(1.75×1.75)=22.86
    """
    
    class People:
        def __init__(self, name, weight, height):
            self.name = name
            self.weight = weight
            self.height = height
    
        # 定义函数的原因1:
        # 1、从bmi的公式上看,bmi应该是触发功能计算得到的
        # 2、bmi是随着身高、体重的变化而动态变化的,不是一个固定的值
        #    说白了,每次都是需要临时计算得到的
    
        # 但是bmi听起来更像是一个数据属性,而非功能
        @property
        def bmi(self):
            return self.weight / (self.height ** 2)
    
    
    obj1 = People('egon', 70, 1.83)
    print(obj1.bmi())
    
    obj1.height=1.86
    print(obj1.bmi())
    #加了@property之后的调用
    print(obj1.bmi)
    
    # 案例二:
    class People:
        def __init__(self, name):
            self.__name = name
    
        def get_name(self):
            return self.__name
    
        def set_name(self, val):
            if type(val) is not str:
                print('必须传入str类型')
                return
            self.__name = val
    
        def del_name(self):
            print('不让删除')
            # del self.__name
    
        name=property(get_name,set_name,del_name)
    
    obj1=People('egon')
    # print(obj1.get_name())
    # obj1.set_name('EGON')
    # print(obj1.get_name())
    # obj1.del_name()
    
    # 人正常的思维逻辑
    print(obj1.name) #
    # obj1.name=18
    # del obj1.name
    
    # 案例二的改进(推荐):
    class People:
        def __init__(self, name):
            self.__name = name
    
    
        @property
        def name(self): # obj1.name
            return self.__name
    
        @name.setter
        def name(self, val): # obj1.name='EGON'
            if type(val) is not str:
                print('必须传入str类型')
                return
            self.__name = val
    
        @name.deleter
        def name(self): # del obj1.name
            print('不让删除')
            # del self.__name
    
    
    obj1=People('egon')
    # 人正常的思维逻辑
    print(obj1.name) #
    # obj1.name=18
    # del obj1.name
    
    
    

    5. 继承

    5.1 什么是继承

    1. 继承是一种创建新类的方式,新建的类可称为子类或派生类,父类又可称为基类或超类,子类会遗传父类的属性

    2. 需要注意的是:python支持多继承
      在Python中,新建的类可以继承一个或多个父类

      ps1: 在python2中有经典类与新式类之分

      新式类:继承了object类的子类,以及该子类的子类子子类。。。

      经典类:没有继承object类的子类,以及该子类的子类子子类。。。

      ps2:

      ​ 在python3中没有继承任何类,那么会默认继承object类,所以python3中所有的类都是新式类

    class Parent1(object):
        x=1111
    
    class Parent2(object):
        pass
    
    class Sub1(Parent1): # 单继承
        pass
    
    class Sub2(Parent1,Parent2): # 多继承
        pass
    print(Sub1.__bases__)
    print(Sub2.__bases__)
    #(<class '__main__.Parent1'>,)
    #(<class '__main__.Parent1'>, <class '__main__.Parent2'>)
    print(Sub1.x)
    
    # print(Parent1.__bases__)
    # print(Parent2.__bases__)
    
    
    1. python的多继承

      优点:子类可以同时遗传多个父类的属性,最大限度地重用代码
      缺点:
      1、违背人的思维习惯:继承表达的是一种什么"是"什么的关系
      2、代码可读性会变差
      3、不建议使用多继承,有可能会引发可恶的菱形问题,扩展性变差,
      如果真的涉及到一个子类不可避免地要重用多个父类的属性,应该使用Mixins

    5.2 为什么要用继承

    用来解决类与类之间代码冗余问题

    5.3 如何实现继承

    # 示范1:类与类之间存在冗余问题
    class Student:
        school='OLDBOY'
        def __init__(self,name,age,sex):
            self.name=name
            self.age=age
            self.sex=sex
        def choose_course(self):
            print('学生%s 正在选课' %self.name)
    
    class Teacher:
        school='OLDBOY'
        def __init__(self,name,age,sex,salary,level):
            self.name=name
            self.age=age
            self.sex=sex
            self.salary=salary
            self.level=level
        def score(self):
            print('老师 %s 正在给学生打分' %self.name)
    
    # 示范2:基于继承解决类与类之间的冗余问题
    class OldboyPeople:
        school = 'OLDBOY'
    
        def __init__(self, name, age, sex):
            self.name = name
            self.age = age
            self.sex = sex
    
    
    class Student(OldboyPeople):
        def choose_course(self):
            print('学生%s 正在选课' % self.name)
    # stu_obj = Student('lili', 18, 'female')
    # print(stu_obj.__dict__)
    # print(stu_obj.school)
    # stu_obj.choose_course()
    
    class Teacher(OldboyPeople):
        #           老师的空对象,'egon',18,'male',3000,10
        def __init__(self, name, age, sex, salary, level):
            # 指名道姓地跟父类OldboyPeople去要__init__
            OldboyPeople.__init__(self,name,age, sex)
            self.salary = salary
            self.level = level
    
        def score(self):
            print('老师 %s 正在给学生打分' % self.name)
    
    tea_obj=Teacher('egon',18,'male',3000,10)
    # print(tea_obj.__dict__)
    # print(tea_obj.school)
    
    tea_obj.score()
    
    

    6. 属性查找

    有了继承关系,对象在查找属性时,先从对象自己的__dict__中找,如果没有则去子类中找,然后再去父类中找

    # 单继承背景下的属性查找
    # 示范一:
    class Foo:
        def f1(self):
            print('Foo.f1')
    
        def f2(self):
            print('Foo.f2')
            self.f1() # obj.f1()
    
    class Bar(Foo):
        def f1(self):
            print('Bar.f1')
    
    obj=Bar()
    obj.f2()
    
    # Foo.f2
    # Foo.f1
    
    # 示范二:
    class Foo:
        def f1(self):
            print('Foo.f1')
    
        def f2(self):
            print('Foo.f2')
            Foo.f1(self) # 调用当前类中的f1
    
    class Bar(Foo):
        def f1(self):
            print('Bar.f1')
    
    obj=Bar()
    obj.f2()
    
    # Foo.f2
    # Foo.f1
    
    # 示范三:
    #父类如果不想让子类覆盖自己的方法,可以采用双下划线开头的方式将方法设置为私有的
    class Foo:
        def __f1(self): # _Foo__f1
            print('Foo.f1')
    
        def f2(self):
            print('Foo.f2')
            self.__f1() # self._Foo__f1,# 调用当前类中的f1
    
    class Bar(Foo):
        def __f1(self): # _Bar__f1
            print('Bar.f1')
    
    obj=Bar()
    obj.f2()
    
    # Foo.f2
    # Foo.f1
    
    

    7. 继承的实现原理

    7.1 菱形问题

    大多数面向对象语言都不支持多继承,而在Python中,一个子类是可以同时继承多个父类的,这固然可以带来一个子类可以对多个不同父类加以重用的好处,但也有可能引发著名的 Diamond problem菱形问题(或称钻石问题,有时候也被称为“死亡钻石”),菱形其实就是对下面这种继承结构的形象比喻

    img

    A类在顶部,B类和C类分别位于其下方,D类在底部将两者连接在一起形成菱形。
    
    

    这种继承结构下导致的问题称之为菱形问题:如果A中有一个方法,B和/或C都重写了该方法,而D没有重写它,那么D继承的是哪个版本的方法:B的还是C的?如下所示

    class A(object):
        # def test(self):
        #     print('from A')
        pass
    class B(A):
        def test(self):
            print('from B')
        pass
    class C(A):
        # def test(self):
        #     print('from C')
        pass
    class D(C,B):
        # def test(self):
        #     print('from D')
        pass
    print(D.mro()) # 类D以及类D的对象访问属性都是参照该类的mro列表
    # obj = D()
    # obj.test()
    
    # print(D.test)
    # print(C.mro()) # 类C以及类C的对象访问属性都是参照该类的mro列表
    # c=C()
    # c.test()
    
    总结:
    类相关的属性查找(类名.属性,该类的对象.属性),都是参照该类的mro
    
    

    7.2 继承原理

    python到底是如何实现继承的呢? 对于你定义的每一个类,Python都会计算出一个方法解析顺序(MRO)列表,该MRO列表就是一个简单的所有基类的线性顺序列表,如下

    >>> D.mro() # 新式类内置了mro方法可以查看线性列表的内容,经典类没有该内置该方法
    [<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>]
    
    

    python会在MRO列表上从左到右开始查找基类,直到找到第一个匹配这个属性的类为止。 而这个MRO列表的构造是通过一个C3线性化算法来实现的。我们不去深究这个算法的数学原理,它实际上就是合并所有父类的MRO列表并遵循如下三条准则:

    1.子类会先于父类被检查
    2.多个父类会根据它们在列表中的顺序被检查
    3.如果对下一个类存在两个合法的选择,选择第一个父类
    
    

    所以obj.test()的查找顺序是,先从对象obj本身的属性里找方法test,没有找到,则参照属性查找的发起者(即obj)所处类D的MRO列表来依次检索,首先在类D中未找到,然后再B中找到方法test

    ps:

    1.由对象发起的属性查找,会从对象自身的属性里检索,没有则会按照对象的类.mro()规定的顺序依次找下去,
    2.由类发起的属性查找,会按照当前类.mro()规定的顺序依次找下去,
    
    

    7.3 深度优先和广度优先

    1 非菱形结构

    img

    class E:
        # def test(self):
        #     print('from E')
        pass
    
    class F:
        def test(self):
            print('from F')
    class B(E):
        # def test(self):
        #     print('from B')
        pass
    class C(F):
        # def test(self):
        #     print('from C')
        pass
    class D:
        def test(self):
            print('from D')
    class A(B, C, D):
        # def test(self):
        #     print('from A')
        pass
    # 新式类
    # print(A.mro()) # A->B->E->C->F->D->object
    
    obj = A()
    obj.test() # 结果为:from F
    
    print(A.mro())
    '''
    [<class '__main__.A'>, <class '__main__.B'>, <class '__main__.E'>, <class '__main__.C'>, <class '__main__.F'>, <class '__main__.D'>, <class 'object'>]
    '''
    # 可依次注释上述类中的方法test来进行验证
    
    

    2 菱形结构

    如果多继承的关系为菱形结构,那么经典类与新式类会有不同MRO,分别对应属性的两种查找方式:深度优先和广度优先

    经典类:不继承object

    ​ 深度优先,会在检索第一条分支的时候就直接一条道走到黑,即会检索大脑袋(共同的父类)

    新式类:继承object

    ​ 广度优先,会在检索最后一条分支的时候检索大脑袋

    img

    class G: # 在python2中,未继承object的类及其子类,都是经典类
        def test(self):
            print('from G')
    class E(G):
        def test(self):
            print('from E')
    class F(G):
        def test(self):
            print('from F')
    class B(E):
        def test(self):
            print('from B')
    class C(F):
        def test(self):
            print('from C')
    class D(G):
        def test(self):
            print('from D')
    class A(B,C,D):
        # def test(self):
        #     print('from A')
        pass
    
    obj = A()
    obj.test() # 如上图,查找顺序为:obj->A->B->E->G->C->F->D->object
    # 可依次注释上述类中的方法test来进行验证,注意请在python2.x中进行测试
    
    

    img

    class G(object):
        def test(self):
            print('from G')
    class E(G):
        def test(self):
            print('from E')
    class F(G):
        def test(self):
            print('from F')
    class B(E):
        def test(self):
            print('from B')
    class C(F):
        def test(self):
            print('from C')
    class D(G):
        def test(self):
            print('from D')
    class A(B,C,D):
        # def test(self):
        #     print('from A')
        pass
    
    obj = A()
    obj.test() # 如上图,查找顺序为:obj->A->B->E->C->F->D->G->object
    # 可依次注释上述类中的方法test来进行验证
    
    

    7 总结

    多继承到底要不用???
    要用,但是规避几点问题
    1、继承结构尽量不要过于复杂
    2、推荐使用mixins机制:在多继承的背景下满足继承的什么"是"什么的关系

    8. Mixins机制:多继承的正确打开方式

    8.1 mixins机制核心:

    就是在多继承背景下尽可能地提升多继承的可读性

    ps:让多继承满足人的思维习惯=》什么"是"什么

    Python提供了Mixins机制,简单来说Mixins机制指的是子类混合(mixin)不同类的功能,而这些类采用统一的命名规范(例如Mixin后缀),以此标识这些类只是用来混合功能的,并不是用来标识子类的从属"is-a"关系的,所以Mixins机制本质仍是多继承,但同样遵守”is-a”关系

    class Vehicle:
        pass
    
    class FlyableMixin:
        def fly(self):
            pass
        
    class CivilAircraft(FlyableMixin,Vehicle):  # 民航飞机
        pass
    
    class Helicopter(FlyableMixin,Vehicle):  # 直升飞机
        pass
    
    class Car(Vehicle):  # 汽车并不会飞,但按照上述继承关系,汽车也能飞了
        pass
    

    8.2 使用Mixin类实现多重继承需要注意的

    • 首先它必须表示某一种功能,而不是某个物品,python 对于mixin类的命名方式一般以 Mixin, able, ible 为后缀
    • 其次它必须责任单一,如果有多个功能,那就写多个Mixin类,一个类可以继承多个Mixin,为了保证遵循继承的“is-a”原则,只能继承一个标识其归属含义的父类
    • 然后,它不依赖于子类的实现
    • 最后,子类即便没有继承这个Mixin类,也照样可以工作,就是缺少了某个功能。(比如飞机照样可以载客,就是不能飞了)

    8.3. 在子类派生的新方法中如何重用父类的功能

    方式一:

    指名道姓调用某一个类下的函数=》不依赖于继承关系

    class OldboyPeople:
        def __init__(self,name,age,sex):
            self.name=name
            self.age=age
            self.sex=sex
    
        def f1(self):
            print('%s say hello' %self.name)
    
    class Teacher(OldboyPeople):
        def __init__(self,name,age,sex,level,salary):
            OldboyPeople.__init__(self,name,age,sex)
    
            self.level = level
            self.salary=salary
    
    tea_obj=Teacher('egon',18,'male',10,3000)
    print(tea_obj.__dict__)
    

    方式二:

    super()调用父类提供给自己的方法=》严格依赖继承关系
    调用super()会得到一个特殊的对象,该对象会参照发起属性查找的那个类的mro

    class OldboyPeople:
        def __init__(self,name,age,sex):
            self.name=name
            self.age=age
            self.sex=sex
    
        def f1(self):
            print('%s say hello' %self.name)
    
    class Teacher(OldboyPeople):
        def __init__(self,name,age,sex,level,salary):
            #Python2中
            # super(Teacher,self).__init__(name,age,sex)
            super().__init__(name,age,sex) # 调用的是方法,自动传入对象
    
            self.level = level
            self.salary=salary
    
    # print(Teacher.mro())
    tea_obj=Teacher('egon',18,'male',10,3000)
    print(tea_obj.__dict__)
    
    

    super()案例

    class A:
        def test(self):
            print('from A')
            super().test()
    
    class B:
        def test(self):
            print('from B')
    
    class C(A,B):
        pass
    
    obj=C()
    obj.test()
    print(C.mro())
    
    from A
    from B
    [<class '__main__.C'>, <class '__main__.A'>, <class '__main__.B'>, <class 'object'>]
    
  • 相关阅读:
    sql当前行数据和之前行数据相加减循环处理
    Sql 查询库、表、列名的语句
    sql 特殊字符替换
    pandas 篇
    JAVA学习--面向对象的特征二:继承性
    JAVA学习--super使用
    JAVA学习--方法的参数传递
    JAVA学习--可变个数的形参的方法
    JAVA学习--面向对象思想的落地法则
    JAVA学习--方法的重载
  • 原文地址:https://www.cnblogs.com/Henry121/p/12662407.html
Copyright © 2020-2023  润新知