一篇文章让你彻底搞清楚 Python 中 self 的含义

刚开始学习 Python 的类写法的时候觉得很是麻烦,为什么定义时需要而调用时又不需要,为什么不能内部简化从而减少我们敲击键盘的次数?
你看完这篇文章后就会明白所有的疑问。

self 代表类的实例,而非类。

实例来说明

class Test:
    def prt(self):
        print(self)
        print(self.__class__)

t = Test()
t.prt()

执行结果如下

<__main__.Test object at 0x000000000284E080>

从上面的例子中可以很明显的看出,self 代表的是类的实例。而 self.__class__则指向类。

self 不必非写成 self

有很多童鞋是先学习别的语言然后学习 Python 的,所以总觉得 self 怪怪的,想写成 this,可以吗?
当然可以,还是把上面的代码改写一下。

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

t = Test()
t.prt()

改成 this 后,运行结果完全一样。
当然,最好还是尊重约定俗成的习惯,使用 self。

self 可以不写吗

在 Python 的解释器内部,当我们调用 t.prt() 时,实际上 Python 解释成 Test.prt(t),也就是说把 self 替换成类的实例。
有兴趣的童鞋可以把上面的 t.prt() 一行改写一下,运行后的实际结果完全相同。
实际上已经部分说明了 self 在定义时不可以省略,如果非要试一下,那么请看下面:

class Test:
    def prt():
        print(self)

t = Test()
t.prt()

运行时提醒错误如下:prt 在定义时没有参数,但是我们运行时强行传了一个参数。
由于上面解释过了 t.prt() 等同于 Test.prt(t),所以程序提醒我们多传了一个参数 t。

Traceback (most recent call last):
  File "h.py", line 6, in 
    t.prt()
TypeError: prt() takes 0 positional arguments but 1 was given

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

class Test:
    def prt():
        print(__class__)
Test.prt()

运行结果如下

 

在继承时,传入的是哪个实例,就是那个传入的实例,而不是指定义了 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 0x0000000002A47080>
<__main__.Child object at 0x0000000002A47080>
<__main__.Parent object at 0x0000000002A47240>

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

在描述符类中,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 0x0000000002A570B8>
self in Desc: <__main__.Desc object at 0x000000000283E208>
<__main__.Desc object at 0x000000000283E208> <__main__.Test object at 0x0000000002A570B8> 

大部分童鞋开始有疑问了,为什么在 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 0x00000000022570B8>
self in Desc: <__main__.Desc object at 0x000000000223E208>
<__main__.Desc object at 0x000000000223E208> None 

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

总结

  • self 在定义时需要定义,但是在调用时会自动传入。
  • self 的名字并不是规定死的,但是最好还是按照约定是用 self
  • self 总是指调用时的类的实例。

以上所有代码在 Python3.4 中均测试通过。
文章转自https://www.cnblogs.com/jessonluo/p/4717140.html侵删

你可能感兴趣的:(一篇文章让你彻底搞清楚 Python 中 self 的含义)