面向对象,动静态方法,名字查找顺序,派生方法

面向对象中级

动静态方法

在类中定义的函数有多种特性
'''
Author:clever-cat
time :2022/11/3-16:26
'''
class Student:
schoo_name = '清华大学'
"""
类中直接定义函数 默认绑定给对象 类调用有几个参数传几个 对象
调用第一个参数就是对象自身
"""
def func1(self):
print('绑定方法')
# 被@classmenthod修饰的函数 默认绑定给类 类调用第一个参数
# 类自身 对象也可以调用并且会自动将产生该对象的类当作第一个参数传入
@classmethod
def func2(cls):
print('我是类方法 classmenthod', cls)
"""
普通函数 无论是类还是对象调用 都必须自己手动传参
"""
@staticmethod
def func3(a):
print('我是静态方法,staticmenthod', a)
st1 = Student()
Student.func1(1231) # 绑定方法
st1.func1() # 绑定方法
st1.func2() # 我是类方法 classmenthod <class '__main__.Student'>
Student.func2() # 我是类方法 classmenthod <class '__main__.Student'>
st1.func3(1231) # 我是静态方法,staticmenthod 1231
Student.func3(1231) # 我是静态方法,staticmenthod 1231

面向对象之继承的概念

面向对象三大特征

  1. 封装

  2. 继承

  3. 多态

    三者中继承最为核心(实操最多 体验最强)
    封装和多态略微抽象

  4. 继承的含义
    在现实生中继承表示人与人之间支援的从属关系
    eg:儿子继承父亲 干女儿继承干爹
    在编程世界中继承表示类与类之间资源的从属关系
    eg:类A继承类B

  5. 继承的目的
    在现实生活中儿子继承父亲就拥有了父亲所有资源的支配权限
    在编程世界中类A继承类B就拥有了类B中所有的数据和方法使用权限

  6. 继承的实操

    class Father:
    name = 'Father'
    pass
    class son(Father):
    name = 'son'
    1.在定义类的时候类名后面可以夹括号填写其它类名 意味着继承其他类
    2.在python中支持多继承 括号内填写多个类名碧池逗号隔开即可
    class F1:
    pass
    class F2:
    pass
    class F3:
    pass
    class Son(F1,F2,F3):
    pass
    1.继承其他类的类 Son
    我们称之为子类、派生类
    2.被继承的类 F1,F2,F3
    我们称之为父类、基类、超类
    我们最常用的就是子类和父类

    继承的本质

    """
    对象:数据与方法的结合体
    类(子类):多个对象相同数据与方法的结合体
    父类:多个类(子类)想要数据与方法的结合体
    类与父类本质都是为了节省代码
    """

    继承本质应该分为两个部分

    1. 抽象:将多个类相同的东西抽出去形成一个新的类
    2. 继承:将多个类继承刚刚抽象出去的新类

名字的查找顺序

  1. 不继承情况下名字的查找顺序

    对象查找名字的顺序
    1.先从对象自己的名称空间中查找
    2.自己没有再去产生该对象的类中查找
    3.如果类中也没有则会报错
    对象 》》》 产生对象的类
    class Son:
    # name = 'Son'
    def func1(self):
    print('Son func1')
    obj = Son()
    # obj.name = 'All'
    # print(obj.name) # All 先找对象名称空间
    # print(obj.name) # All 先找对象名称空间 #Son 在查找实例化对象类中的命名空间
    # obj.name = 'All'
    print(obj.name) # AttributeError: 'Son' object has no attribute 'name'
  2. 单继承情况下名字的查找顺序、
    对象自身》》产生对象的类》》父类

    class A:
    # name = 'A'
    pass
    class B(A):
    # name = 'B'
    pass
    class C(B):
    # name = 'C'
    pass
    # obj = C()
    # obj.name = 'ALL'
    # print(obj.name) # ALL
    # print(obj.name) # C
    # print(obj.name) # B
    # print(obj.name) #A
    # 查找不到直接报错
    # print(obj.name) # AttributeError: 'C' object has no attribute 'name'
    # 烧脑小案例
    class AA:
    def func1(self):
    print('我是AA func1')
    def func2(self):
    print('我是AA func2')
    self.func1()
    class AB(AA):
    def func1(self):
    print('我是AB func1')
    obj = AB()
    obj.func2()
    # 我是AA func2
    # 我是AB func1
    """
    1.首先查找自己,没有,去产生对象的类中查找没有
    2.再去父类中查找,查早到,调用,打印 我是AA func2 继续执行,self.func1()
    3.重点看self,类中的绑定方法,谁调用谁是主人公,说要就是obj对象
    4.从obj对象名称空间开始查找
    5.在产生对象的类中找到了,执行代码。打印 我是AB func1
    """
  3. 多继承情况下名字查找顺序

    1. 菱形继承

      ​ 广度优先搜索(最后才会找闭环的顶点)

    2. 非菱形继承

      ​ 深度优先搜索(从左往右每条道走完为止)

    3. mro()方法了可以直接获取名字的查找顺序

    对象自身》》产生对象的类》》父类(从左往右)

    class F1:
    name = 'jason'
    pass
    class F2:
    # name = 'oscar'
    pass
    class F3:
    # name = 'jerry'
    pass
    class S1(F1, F2, F3):
    # name = '嘿嘿嘿'
    pass
    obj = S1()
    # obj.name = '想干饭'
    print(obj.name)
    '''
    对象自身 >>> 产生对象的类 >>> 父类(从左往右)
    '''
    class G:
    name = 'from G'
    pass
    class A:
    # name = 'from A'
    pass
    class B:
    # name = 'from B'
    pass
    class C:
    # name = 'from C'
    pass
    class D(A):
    # name = 'from D'
    pass
    class E(B):
    # name = 'from E'
    pass
    class F(C):
    # name = 'from F'
    pass
    class S1(D,E,F):
    pass
    obj = S1()
    # print(obj.name)
    print(S1.mro())

经典类与新式类

  1. 经典类:不继承object或者其子类的类

  2. 新式类:继承object或者其子类的类

    • 在python2中有经典类与新式类
    • 在python3中只有新式类(所有类默认继承object)

    class Student(object):pass

    以后在定义的时候 如果没有其他父类继承 也可以习惯性的写上可以增加兼容python2

派生方法

# 子类基于父类的方法做了扩展
class Father(object):
def __init__(self, name, age, pwd):
self.name = name
self.age = age
self.pwd = pwd
class Son(Father):
def __init__(self, name, age, pwd, gender):
# 子类调用父类的方法
super().__init__(name, age, pwd)
self.gender = gender
obj = Son('张三', 18, '123', 'male')
print(obj.name) # 张三
print(obj.gender) # male

super()方法使用注意事项

class E:
def a(self):
print('class E')
pass
class A:
def a(self):
print('class A')
pass
class B(A):
def a(self):
print('class B')
pass
class C(B):
def a(self):
print('class C')
super().a() # class B
super(C, self).a() # class B 正常情况下
super(B, self).a() # class A 可以指定重哪个类想向上一级调用父类方法
# super(A, self).a() # 可以指定从哪个类想向上一级调用父类方法,如果父级没有则报错
# super(E, self).a() # 可以指定从哪个类想向上一级调用父类方法,如果这个类不是本类的父类则报错
pass
obj = C()
obj.a()
print(C.__bases__)
print(C.__base__)
print(C.mro())

本文作者:clever-cat

本文链接:https://www.cnblogs.com/clever-cat/p/16855526.html

版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。

posted @   clever-cat  阅读(35)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧
· 园子的第一款AI主题卫衣上架——"HELLO! HOW CAN I ASSIST YOU TODAY
点击右上角即可分享
微信分享提示
评论
收藏
关注
推荐
深色
回顶
收起