python-深入类和对象

1.鸭子类型与多态

鸭子类型
多态的概念是应用于Java和C#这一类强类型语言中,而Python崇尚"鸭子类型"。
动态语言调用实例方法时不检查类型,只要方法存在,参数正确,就可以调用。这就是动态语言的“鸭子类型”,它并不要求严格的继承体系,一个对象只要“看起来像鸭子,走起路来像鸭子”,那它就可以被看做是鸭子。

#鸭子类型 在运行之前 Cat,Dog都是在列表里面,当作变量
#当运行时,加上()调用info() 才明确Cat是一个类
class Cat(object):
    def info(self):
        print("i am cat")
class Dog(object):
    def info(self):
        print("i am dog")
class Duck(object):
    def info(self):
        print("i am duck")
animal_list=[Cat,Dog,Duck]
for a in animal_list:
    a().info()

多态
定义时的类型和运行时的类型不一样,就称为多态。

2.抽象基类(abc模块)

抽象基类介绍
抽象基类(abstract base class,ABC):抽象基类就是类里定义了纯虚成员函数的类。纯虚函数只提供了接口,并没有具体实现。抽象基类不能被实例化(不能创建对象),通常是作为基类供子类继承,子类中重写虚函数,实现具体的接口。
抽象基类就是定义各种方法而不做具体实现的类,任何继承自抽象基类的类必须实现这些方法,否则无法实例化。

class Demo(object):
    def __init__(self,li):
        self.li=li
    def __len__(self):
         return len(self.li)
i=["B","狐狸","狼"]
d=Demo(i)
print(d) #<__main__.Demo object at 0x0000020FE8781AC8>
print(len(d)) # 会触发__len__  return len(self.li)  所以 返回3
#如何判断 Demo中 是否 含有 __len__魔法方法
print(hasattr(d,"__len__"))# 判断 d的内部 是否含有__len__   返回值为 布尔 True
"""
可以通过判断 d 是否是 Sized的子类  然后进一步判断 d这个对象的类 是否 含有 __len__方法
"""
from collections.abc import Sized
# 判断d 是否是 Sized 这个类
print(isinstance(d,Sized))    # True 说明d 是Sized 类型

抽象基类应用场景
1.我们去检查某个类中是否有某种方法

class Demo(object):
    def __init__(self,elist):
        self.elist = elist
    def __len__(self):
        return len(self.elist)
d = Demo(["oldAmy", "ls"])
print(len(d))
from collections.abc import Sized
print(isinstance(d, Sized))        # True  d是Sized

查看Sized源码
python安装路径下Lib中的隐藏文件_collections_abc.py

class Sized(metaclass=ABCMeta):
    __slots__ = ()
    @abstractmethod
    def __len__(self):
        return 0
    @classmethod   
    def __subclasshook__(cls, C):
        if cls is Sized:
            return _check_methods(C, "__len__")
        return NotImplemented

注意:
注册的虚拟子类不会继承抽象基类的任何方法和属性
2.我们需要强调某个子类必须实现某些方法
例子:
定义父类Cache
封装CRUD方法。强制子类重写该方法。
定义子类Redis

#方法1 主动抛出异常
class CacheBase(object):
    def dele(self):
        raise NotImplementedError
    def crea(self):
        raise NotImplementedError
class RedisBase(CacheBase):
    """
    1.子类 如果不重写 父类 方法 访问时  直接抛出异常
    """
    def crea(self):
        print("create")
r = RedisBase()
r.crea()
r.dele()
"""
强制子类重写
方法2 抽象基类实现
"""
import abc
# 注意:不是直接继承object 而是继承abc.ABCMeta
class CacheBase(metaclass=abc.ABCMeta):
    @abc.abstractmethod
    def dele(self):
        pass
    def crea(self):
        pass
class RedisBase(CacheBase):
    def dele(self):
        pass
r = RedisBase()
"""
抽象基类 : 实例化的时候检测
主动抛出异常 :调用时才会检测
"""

3.type与isinstance区别

type 不考虑 继承关系
isinstance 考虑继承关系

a=1
b='浩子'
print (isinstance(b,(int,str))) # 返回值为布尔值  ()-->or
print(type(a))#   int
class Father(object):
    pass
class Son(Father):
    pass
ls=Son()
print(isinstance(ls,Son)) # True
print(isinstance(ls, Father)) # True  继承
print(type(ls) is Son) # True
print(type(ls) is Father) # False 并不考虑继承关系 主要 两个类
print(f"Son_id:{id(Son)},Father_id:{id(Father)}")

4.类属性与实例属性

基本查找顺序
对象是可以向上查找的,所以可以访问到类属性
当对象自己有该实例属性时 ,则输出的是自己的
类不能向下查找,所以只能访问到类属性
多继承查询顺序
继承关系如下,则属性查找顺序为
python-深入类和对象_第1张图片
实际上,python2.2(金典类)之前的算法:MRO算法,DFS(deep first search) 深度优先。
python-深入类和对象_第2张图片
如下图,菱形继承,执行顺序如何?
python-深入类和对象_第3张图片
在python2.2版本之后,引入BFS(广度优先)。
在python新式类,就引入了C3算法,通过className.__mro__来查看。
python-深入类和对象_第4张图片

class Father(object):
    # 类属性
    cls_ins = 1
    def __init__(self, a, b):
        # 实例属性
        self.a = a
        self.b = b
        # self.cls_ins = 886
zs = Father(3, 4)
print(zs.cls_ins, zs.a, zs.b)  # 1 3 4
print(Father.cls_ins)
# print(Father.a)  # 报错
"""
对象 可以向上查找 可以访问到 类属性
类不能向下查找 所以 只能访问到类属性。 类名.实例属性  报错
"""
Father.cls_ins = 778  # 覆盖
zs.cls_ins = 886  # 相当于封装
print(Father.cls_ins)
print(zs.cls_ins)
class MyTest(object):
    ins = "cls_ins"
    def __init__(self):
        self.ins = "ins"
t = MyTest()
print(MyTest.ins)  # cls_ins
print(t.ins)  # ins
"""
当对象 自己有该实例属性的时候  就直接输出自己的
当对象 自己没有该属性的时候 才会向类 向上查找 
"""
# 棱形继承
class D(object):
    pass
class B(D):
    pass
class C(D):
    pass
class A(B, C):
    pass
print(A.__mro__)#(, , , , )

5.Python对象自省机制

自省是通过一定的机制查询到对象的内部结构
Python中比较常见的自省(introspection)机制(函数用法)有: dir(),type(), hasattr(), isinstance(),通过这些函数,我们能够在程序运行时得知对象的类型,判断对象是否存在某个属性,访问对象的属性。

class Perpon(object):
    name = "会太阳"
class Student(Perpon):
    def __init__(self, school_name):
        self.school_name = school_name
hty = Student("Logic_edu")
print(hty.__dict__)  # {'school_name': 'Logic_edu'}  当前对象的属性 {"属性":"属性的值"}
print(dir(hty))  # []  考虑到继承的成员
#{'school_name': 'Logic_edu'}
['__class__', '__delattr__', '__dict__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__init_subclass__', '__le__', '__lt__', '__module__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'name', 'school_name']

6.super函数

# """
# super基础使用
# """
# class A(object):
#     def __init__(self):
#         print("A")
# class B(A):
#     def __init__(self):
#         print("B")
#         # 重写父类的构造方法  再调用父类的方法
#         super().__init__()
# b = B()
"""
软编码与硬编码
"""
# class Person(object):
#     def __init__(self, name, age, weight):
#         self.name = name
#         self.age = age
#         self.weight = weight
#     def speak(self):
#         print(f"{self.name}说:我{self.age}岁")
# class Student(Person):
#     def __init__(self, name, age, weight, grade):
#         self.grade=grade
#         # 软编码 -> 父类的名字可以随便更改
#         super().__init__(name, age, weight)
#         # 硬编码
#         # Person.__init__(self, name, age, weight)
#         # self.grade = grade
#     def speak(self):
#         print(f"{self.name}说:我{self.age}岁,我在{self.grade}年级,我体重{self.weight}")
# hz = Student('hz', 20, 120, 2)
# hz.speak()
 # super函数的继承机制
class A(object):
    def __init__(self):
        print("A")
class C(A):
    def __init__(self):
        print("B")
        super().__init__()
class B(A):
    def __init__(self):
        print("C")
        super().__init__()
class D(B,C):
    def __init__(self):
        print("D")
        super().__init__()
if __name__ == '__main__':
    d = D()
    print(D.__mro__)

你可能感兴趣的:(Python)