`
uule
  • 浏览: 6310493 次
  • 性别: Icon_minigender_1
  • 来自: 一片神奇的土地
社区版块
存档分类
最新评论

Python中self的含义

 
阅读更多

 Python为什么要Self?

 

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

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

实例来说明 

class Test:
    def prt(self):
        print(self)
        print(self.__class__)
 
t = Test()
t.prt()
 结果如下 
<__main__.Test object at 0x000000000284E080>
<class '__main__.Test'>
 从上面的例子中可以很明显的看出,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 <module>
    t.prt()
TypeError: prt() takes 0 positional arguments but 1 was given

 

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

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

 运行结果如下

<class '__main__.Test'>

 

在继承时,传入的是哪个实例,就是那个传入的实例,而不是指定义了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> <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 0x00000000022570B8>
self in Desc: <__main__.Desc object at 0x000000000223E208>
<__main__.Desc object at 0x000000000223E208> None <class '__main__.Test'>

 

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

 

总结

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

相关推荐

    关于Python中定制类的比较运算实例

    Python中的比较运算有几种:小于、小于等于、等于、大于等于、大于、不等于等。如果我们的数据对象具有明确的物理含义,比如说数值是带有数字与物理单位的字符串组合,那么进行大小比较的时候就可以做此定制。 下面...

    python @property的用法及含义全面解析

    在接触python时最开始接触的代码,取长方形的长和宽,定义一个长方形类,然后设置长方形的长宽属性,通过实例化的方式调用长和宽,像如下代码一样。 class Rectangle(object): def __init__(self): self.width =...

    Python反射用法实例简析

    本文实例讲述了Python反射用法。分享给大家供大家参考,具体如下: class Person: def __init__(self): self.name = "zjgtan" def getName(self): return self.name 反射的简单含义: 通过类名获得类的实例...

    Python面向对象特殊属性及方法解析

    Python中对象包括很多双下划綫开始和结束的属性,这些特殊属性,有特殊用法。 特殊方法 含义 obj.__dict__ 对象的属性字典 obj.__class__ 对象所处的类 class.__bases__ 类的基本元组(多继承) class.__...

    简单分析python的类变量、实例变量

    实例变量是定义在方法中的变量,使用self绑定到实例上的变量,只是对当前实例起作用。 2、访问 类变量 在类的内部和外部类变量都可以直接使用className.类变量的形式访问。但是在类的内部,也可以使用self.类变量...

    零基础小白的python学习笔记之类和类对象(一)基本定义

    1.类对象 =属性+方法 class Girl: #属性 name = “vv” #方法 ...3.Python里多态的含义:不同的类调用名字相同的函数,其实现方法不同 class A: def fun(self): print('我是A') class B: def

    基于遗传算法、强化学习自动选择高频因子python实现源码+项目说明.zip

    (2) 因子开发的格式 因子名称: Liquidity_DeltaTurnover 一级分类: PriceVolume 二级分类: Liquidity 因子编号: Factor01040009 因子含义: 因子说明: 因子算法: 因子来源:海通证券《选股因子系列研究 *****》 ...

    Python 类的魔法属性用法实例分析

    无论人或事物往往都有不按套路出牌的情况,Python的类属性也是如此,存在着一些具有特殊含义的属性,详情如下: 1. doc 表示类的描述信息 class Foo: 描述类信息,这是用于看片的神奇 def func(self): pass ...

    CS_Python_Basic

    文章目录PythonpipPython函数和语法excel获取数据类型Class(object):中object的作用if __name__ == ‘__main__’: 的作用zip()enumerate() 方法with语法self的含义切片...Python pip pip list 查看安装了哪些包 ...

    Python基于Nonebot2的修仙文字游戏插件+使用说明.zip

    2、子插件的配置会在插件运行后在子插件文件中生成config.json文件,该文件字段含义在同级目录的xxxconfig.py有备注。注意:修改配置只需要修改json即可,修改.py文件的话需要删除json文件才会生效,任何修改都需要...

    《Python机器学习基础教程》学习记录(三)

      k 近邻算法中 k 的含义是,我们可以考虑训练集中与新数据点最近的任意 k 个邻居(比如 说,距离最近的 3 个或 5 个邻居),而不是只考虑最近的那一个。然后,我们可以用这些 邻居中数量最多的类别做出预测。 #!/...

    python实现一个简单的并查集的示例代码

    用UnionFind类来表示一个并查集,在构造函数中,初始化一个数组parent,parent[i]表示的含义为,索引为i的节点,它的直接父节点为parent[i]。初始化时各个节点都不相连,因此初始化parent[i]=i,让自己成为自己的父...

    SelfBot:一个不和谐的自我机器人,可以做很多事情

    自举一个不和谐的自我机器人,可以做很多事情确保在代码上方填写您的信息含义标记和前缀

    ruigi

    瑞吉日语中的Ruiji含义相似:Ruigi〜Luigi 该名称封装了此程序包的本质,与Luigi非常相似,但不完全相同。 管道是: 很容易开始易于配置和重复使用配方使每个程序员都成为数据管道专家就像keras是为tensorflow制作的...

    Transformers:变压器的 Pytorch 实现

    现在的一个问题是,同一个词在不同的句子中可以有不同的含义,这就是位置编码进入的地方。 由于transformers不包含递归和卷积,为了让模型利用序列的顺序,它必须利用一些关于单词在序列中的相对或绝对位置的信息。...

Global site tag (gtag.js) - Google Analytics