在python类中的各个方法(包括构造方法__init__())的参数里都要写self,下面用实验来探究self的用法,从而体会其含义。
一言以蔽之
self代表“本实例”,很像java里的this,但用法又稍微不太一样。
1. 直接定义在类里的成员变量不能加self
class MyClass:
"""class"""
i = 3
self.h = 5
错误信息:
Traceback (most recent call last):
File “D:*******************.py”, line 1, in <module>
class MyClass:
File “D:*******************.py”, line 4, in MyClass
self.h = 5
NameError: name ‘self’ is not defined
2. 内部方法中不能直接使用没有self直接定义的变量(要用需加self):
class MyClass:
"""class"""
i = 3
def __init__(self):
self.j = 5
def f(self):
print(self.i) #可以正常输出
print(self.j) #可以正常输出
print(i) #报错
obj = MyClass()
obj.f()
结果与错误信息(未找到i的定义):
3
5
Traceback (most recent call last):
File “D:******************.py”, line 13, in <module>
obj.f()
File “D:******************.py”, line 9, in f
print(i)
NameError: name ‘i’ is not defined
3. 类的内部函数的参数一定要写self:
class MyClass:
"""class"""
i = 3
def f(x):
print(x)
obj = MyClass()
obj.f(x) # 调用内部函数时会报
错误信息:
Traceback (most recent call last):
File “D:******************.py”, line 8, in <module>
obj.f(1)
TypeError: f() takes 1 positional argument but 2 were given
错误中说函数f()只能接受1个参数,而调用时给了两个。
嗯?等等……什么鬼?我们调用的时候是只给了一个参数x呀,怎么会有第二个参数呢?
原来,python会把obj.f(x)看作Myclass.f(obj, x),即会把调用函数的对象看作一个参数隐式地传给函数。
4. self参数一定得是内部函数的第一个参数
改写3.中的调用语句:
class MyClass:
"""class"""
i = 3
def f(x):
print(x)
print(x.i)
obj = MyClass()
obj.f()
结果:
<__main__.MyClass object at 0x…>
3
可见,这里的x是一个对象,我们甚至能看到x里面的成员变量,也就是说,x完全充当了self的角色。另外也可以看出,self并不是python的强制写法,它可以写作任意的名字,但就是不能缺省。
5. 修改并重做代码2
这次我们在类外定义一个i
class MyClass:
"""class"""
i = 3
def f(self):
print(self.i)
print(i)
i = 10086
obj = MyClass()
obj.f(
结果:
3
10086
由3. 可知python把obj.f()看作了Myclass.f(obj),也就是在i=10086这条语句下方执行了一行叫做Myclass.f(obj)的代码,此时,不带self的i自然指的应该是定义在函数执行语句之前的i,所以print(i)的结果是10086,而且这也解释了之前2. 里报的错误原因(未找到i,也就是说在调用函数之前的语句中没有找到i的定义)
启示:类外与类内的参数名称应尽可能不重名,尤其注意与内部方法里面的局部参数不要重名,因为方法调用局部变量与类外定义的变量都是不加self的。