• python self


    1. self代表类的实例,而非类

    class Test:
        def prt(self):
            print(self)
            print(self.__class__)
            
    t = Test()
    t.prt()
    
    <__main__.Test object at 0x000002345ED1DFC8>
    <class '__main__.Test'>
    

    上面的例子,self 代表的是类的实例。而 self.class 则指向类。



    2. self不必写成self

    学过其他语言的,觉得 self 怪怪的,想写成 this,也是可以的。

    class Test:
        def prt(this):
            print(this)
            print(this.__class__)
            
    t = Test()
    t.prt()
    

    改成 this,运行结果一样。但最好还是尊重约定俗成的习惯。



    3. self可以不写吗

    在 python 中,当我们调用 t.prt() 时,实际上 python 解释成 Test.prt(t),也就是说把 self 替换成类的实例。

    class Test:
        def prt():
            print(self)
            
    t = Test()
    t.prt()
    

    运行时出错,因为 prt 没有参数,我们强行传了一个参数,t.prt() 等同于Test.prt(t)

    TypeError                                 Traceback (most recent call last)
    <ipython-input-2-282f362e7e8c> in <module>
          4 
          5 t = Test()
    ----> 6 t.prt()
    
    TypeError: prt() takes 0 positional arguments but 1 was given
    

    当然,我们可以定义和调用时均不传类的实例,这就是类方法。

    class Test:
        def prt():
            print(__class__)
    Test.prt()
    
    <class '__main__.Test'>
    

    4. 继承时,传入的是哪个实例,就是那个传入实例,而不是指定义了self的类的实例

    class Parent:
        def pprt(self):
            print(self)
     
    class Child(Parent):
        def cprt(self):
            print(self)
    c = Child()
    c.cprt()
    c.pprt()
    p = Parent()
    p.pprt()
    
    <__main__.Child object at 0x0000023460235C48>
    <__main__.Child object at 0x0000023460235C48>
    <__main__.Parent object at 0x0000023460235C88>
    

    运行 c.cprt() 时,指的是 Child 类的实例。

    但是在运行 c.pprt()时,等同于 Child.pprt(c),所以 self 指的依然是 Child 类的实例,由于 self 中没有定义 pprt() 方法,所以沿着继承树往上找,发现父类 Parent 中定义了 pprt() 方法,所以就会调用。



    5. 在描述符类中,self指的是描述符类的实例

    class Desc:
        def __get__(self, ins, cls):
            print('self in Desc: %s' % self)
            print(self, ins, cls)
            
    class Test:
        x = Desc()
        def prt(self):
            print('self in Test: %s' % self)
    
    t = Test()
    t.prt()
    t.x
    
    self in Test: <__main__.Test object at 0x000002346022D308>
    self in Desc: <__main__.Desc object at 0x000002346022D2C8> 
    <__main__.Desc object at 0x000002346022D2C8> <__main__.Test object at 0x000002346022D308> <class '__main__.Test'>
    

    为什么在 Desc 类中定义的 self 不是应该调用它的实例 t 吗?怎么变成了 Desc 类的实例了?

    注意:这里调用的是 t.x,也就是说 Test 类的实例 t 的属性 x,由于实例 t 中没有定义属性 x,所以找到类属性 x,而该属性是描述符属性,为 Desc 类的实例而已,所以此处没有顶用 Test 的任何方法。

    如果直接通过类来调用属性 x 也可以得到相同的结果。

    下面是把 t.x 改为 Test.x 运行的结果。

    self in Test: <__main__.Test object at 0x00000234602280C8>
    self in Desc: <__main__.Desc object at 0x0000023460228388> 
    <__main__.Desc object at 0x0000023460228388> None <class '__main__.Test'>
    

    题外话:由于很多时候描述符类中仍然需要知道调用该描述符的实例是谁,所以在描述符类中存在第二个参数 ins,用来表示调用它的类实例,所以 t.x 可以看到第三行中的运行结果中第二项 <_main_.Test object at 0x000002346022D308>。而采用 Test.x进行调用时,没有实例,返回None。



    总结

    • self 在定义时需要定义,但是在调用时会自动传入
    • self 的名字并不是规定死的,但最好还是按照约定使用 self
    • self 总是指调用时的类的实例。
  • 相关阅读:
    Ansible跳板机自动部署
    nginx展示文件目录
    【转】消息钩子注册浅析
    windows临界区
    windbg定位死锁
    Windows工作集内存
    我的spring boot,杨帆、起航!
    CursorFileManager对cursor文件的读写
    eclipse执行maven install命令时跳过test
    bASE--Risk
  • 原文地址:https://www.cnblogs.com/keye/p/15099041.html
Copyright © 2020-2023  润新知