Python中self的解释

定义

首先我们要搞明白Python的类中为要什么要用到self这个单词呢,为什么不用"zhangsan"、"lisi"这样的名字呢,这可定有他的用意。查Google翻译解释:

self
名词的解释:自;自我;自己

那么问题来了,这里的自己指代的到底是谁,下面听我详细分解。(仅属于自己的见解,有不对的地方,还望不吝赐教)

分析

测试代码以一下这个代码及其它的扩充为基础,先贴代码,为了好理解,别说了半天不知道我在说什么。

class A():
    def song(one):
        print(one)
    def sing(self,one):
        print(one)
1、首先要搞清楚的一个问题 a=A()与a=A,这两个赋值之间的区别
a=A()
print(a)
a=A
print(a)
print(A)
结果如下:
<__main__.A object at 0x0000020A69C81780>
<class '__main__.A'>
<class '__main__.A'>
分析:
a=A()  是有地址的,这说明他是一个实体,比如 int a  中的 a 一样
a=A    是没有地址的,发现他和A打印的结果一毛一样,这说明他就是A,也就是一个类名,
       等价于上一句中 int a 中的 int
       
那么也就是说下面这两语句是等价的
b=A()   与   b=a()
-------------------------------------------------------------------------
通过上面的分析,我们知道,a=A()相当于把 A 给实例化了,什么叫实例化呢,类比与 int a ,
int 什么都不能做,即 int 不能被赋值,不能进行算术操作等等,但是 a 可以(其实这里的 A 与 int
是一样的,追溯到C语言,类只不过是一种自定义的结构变量而已,只不过功能更加强大)。
那具体怎么个实例发呢?
2、self 到底指代的是谁
class A():
    def song(one):
        print(one)
    def sing(self,one):
        print(one)
a=A()
A.sing(a,"hello")  #注意这条语句
a.sing("hello")
打印结果:
hello
hello
-------------------------------
A.sing("hello")
错误信息:
TypeError: sing() missing 1 required positional argument: 'one'
提示没有给 one 赋值
--------------------------------
a.song("hello")
错误信息:
TypeError: song() takes 1 positional argument but 2 were given
那么问题来了,多给了一个什么参数,为什么会多给呢?
--------------------------------
把类A改一下
class A():
    def song(one,two):   #注意这里多给song了一个参数
        print(one)
a.song("hello")   //这里的参数  "hello"赋给了 two
print(a)
打印结果:
<__main__.A object at 0x0000020A69C39CC0>
<__main__.A object at 0x0000020A69C39CC0>
是不是很奇怪,为什么a.song("hello")返回的是  实例  a  的地址呢?

从这个例子中你就可以发现,实际上 a.sing(“hello”) 等价于 A.sing(a,“hello”),而self就是实例 a 自己,而且你发现 self ,也可以用别的单词来代替。
通过这段分析,我们简单的知道了,类中的方法第一个参数必须是 self ,不然实例无法正确调用类中的方法,也就是说,如果方法中第一个参数不是 self(广义的),那么这个方法是没有任何价值的,因为实例无法调用它,一个无法被调用的方法真不知道有什么用。

参数前面的 self

x=6
class A():
    def sing(self):
        self.x=10
    def mutl(self):
        y=10*x
        print(y)
a=A()  
a.mutl()

这里先猜一下结果,到底会打印 60 呢,还是 100 呢?要弄清这个问题,就要先弄清 mutl() 中的 x 到底是哪一个,先看结果在分析。
结果是:60
为什么不调用类内部的 x 参数而跑去调用类外的 x 呢?还是那个问题,self 到底指代的是谁,self 就是 a 本身,那么问题就很明显了,mutl() 方法中的 x 前面没有加 self 所以他调用的不是实例(注意这里说的是实例,而不是类)自身的参数。
到这里我想你大概明白了,参数前面有self和没self的区别了,简单说,带self的参数是人家实例自身的,不带self的,爱谁谁,实例不管。

不同方法中的参数是否可以互相使用

class A():
    def mutl(self):
        x=5
    def sing(self):
        y=5*x
        print(y)
a=A()  
a.mutl()
a.sing()
错误如下:
<ipython-input-3-e305f6d63af0> in sing(self)
      3         x=5
      4     def sing(self):
----> 5         y=5*x
      6         print(y)
      7 #     def print1(self):

NameError: name 'x' is not defined

可以看到,对于参数前面不加self的参数,只适用于本方法,不能跨方法调用。


class A():
    def mutl(self):
        self.x=5
    def sing(self):
        y=5*self.x
        print(y)
a=A()  
a.mutl()
a.sing()
打印结果:
25

这里可以清晰地看到不在报错,加了self,立马实现跨方法调用。


class A():
    def mutl(self):
        self.x=5
    def sing(self):
        y=5*self.x
        print(y)
a=A()  
a.sing()

先思考一下这段代码与上面的不同之处。
是不是这里没有调用方法 a.mutl()呢,那结果是什么呢,思考一下。

<ipython-input-6-ddb67bd4703e> in sing(self)
      3         self.x=5
      4     def sing(self):
----> 5         y=5*self.x
      6         print(y)
      7 

AttributeError: 'A' object has no attribute 'x'

显示类中没有参数 x ,我什么呢,我明明定义了啊,对滴。但是在把类实例之后你没有通过方法 a.mutl() 来实现对 self.x 进行定义。这也就是为什么要先有构造函数了,先把所有类中的参数全部定义以后,你才能使用。

小结

其实说了这么多,说的也很乱,但是我想说明的就两点:

  1. 类中的方法第一个参数是 self 的才可以被实例调用。
  2. 类中带 self 的参数都是 实例 的,实例对这个参数拥有所有权,即实例中所有的方法都可以使用实例的参数。

至于参数什么时候用加self什么时候不用加self,我想你只要弄白了self到底起什么作用,这个问题就不说自明了。

由于水平有限,分析中可能包含一些错误,还望各位不吝赐教,多多海涵!

  • 207
    点赞
  • 608
    收藏
    觉得还不错? 一键收藏
  • 28
    评论
评论 28
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值