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 总是指调用时的类的实例。
posted @ 2021-08-04 15:20  做梦当财神  阅读(100)  评论(0编辑  收藏  举报