类成员:
    # 字段
        - 普通字段,保存在对象中,执行只能通过对象访问
        - 静态字段,保存在类中,  执行 可以通过对象访问 也可以通过类访问
       
    # 方法
        - 普通方法,保存在类中,由对象来调用,self=》对象
        - 静态方法,保存在类中,由类直接调用,对象也可以调用
        -   类方法,保存在类中,由类直接调用,cls=》当前类,对象也可以调用

class Foo:
    nation = 'china'# 静态字段
    def __init__(self, name):
        self.name = name  # 普通字段
    
    def show(self):   # 普通方法
        print(self.name)

    @staticmethod   # 静态方法
    def stat(a1,a2):
        print(a1,a2)

    @classmethod    # 类方法
    def classmd(cls):# cls 是类名
        print(cls)
        print('classmd')

obj = Foo('pis')
obj.name
obj.show()
#Foo.show(obj)#也能成功调用,传入对象参数 Foo.stat(
1,2)#类直接调用 Foo.classmd()#类直接调用
obj.stat(1,2) #对象直接调用
obj.classmd()#对象直接调用

 

python并没有真正的私有属性(c++有public protect private)

        实际上:用__定义的属性,只是被改名换姓而已。

       用_定义的属性,意义在于唤起用户的注意,看成私有属性.实际上只是普通方法。

        表现形式:双下划线的函数或属性,在类定义中(类中方法)可以调用和访问,类的实例不可以直接访问,子类不可访问。

                          单下划线的函数或属性,在类定义中(类中方法)可以调用和访问,类的实例可以直接访问,子类中可以访问;

        对于双下划线的函数或属性,Python解释器使用了名字混淆的方法, 将私有的方法"__method"变成了"_classname__method"了。

class Base(object):#object 为基类,有一些常用的方法
    def __private(self):
        print("private value in Base")

    def _Base__private(self):
        print("_Base__private value in Base")#如果没有定义,则调用上一方法

    def _protected(self):  #已被覆盖
        print("protected value in Base")

    def public(self):
        print("public value in Base")
        self.__private()
"""此时调用父类的方法-----------原因如下:
    self.__private()方法调用时先从子类寻找,再从父类寻找,没有则报错。
    如果是子类的普通方法调用,虽然子类的私有方法已改名,但能识别,而调用子类的私有方法。
    现在父类的public调用,而子类的私有方法已改名,不能识别。
    父类的public方法只能寻找并调用父类的__private()方法。
    我们知道子类的普通方法能够调用子类的私有方法(虽然已改名),父类也是。所以能够找到父类的私有方法。
    那么父类的私有方法改名了吗?我们加入代码测试------已改
"""
        self._protected()#已被覆盖

class Derived(Base):
    def __private(self):
        print("override private")

    def _protected(self):
        print("override protected")

d = Derived() #创建对象
d.public() #继承父类的方法
d._protected()#调用子类的方法
d._Derived__private()#调用子类__private(),不推荐使用。不符合初衷。

#d.__private()#已改名,不能调用。父类就算有相同方法根本看不到。

结果如下:

public value in Base
_Base__private value in Base
override protected
override protected
override private

 

python 继承

1. 多重继承时方法查找顺序如何呢?

       类都有一个名为 __mro__ 的属性, 它的值是一个元组, 按照方法解析顺序列出各个超类, 从当前类一直向上, 直到object 类。 D 类的 __mro__ 属性如下 :

(<class '__main__.D'>, <class '__main__.B'>, <class '__main__.C'>, <class '__main__.A'>, <class 'object'>)

通俗点也就是三大原则:a. 左侧优先
                                        b. 一条道走到黑
                                        c. 同一个根时,根最后执行

 

2. 单继承和多继承中构造函数__init__

    单继承情况

    (1)子类默认无__init__时,则会直接继承父类__init__

    (2)子类中含有__init__时,不会自动调用父类__init__,如需使用父类__init__中的变量,则需要在子类__init__中显式调用

     多继承情况
    (1)子类继承于多个父类,并且子类无__init__时,则按继承顺序(MRO),哪个父类在最前面且有自己的__init__,则继承它;若最前面的父类无__init__,则继承第二个父类的__init__,若还是无__init__,则依次往后寻找,直到继承的某个父类含__init__。

    (2)子类继承于多个父类,并且子类含__init__时,和单继承的(2)类似,不会自动调用所有父类__init__,如需使用某个父类__init__中的变量,则需要在子类__init__中显式调用