self是什么
在Python类中规定,在定义类的过程中,无论是显式创建类的构造方法,还是向类中添加实例方法,都要求将 self 参数作为方法的第一个参数,表示当前类的对象,可以调用当前类中的属性和方法。
self代表类的实例,不是类本身
class Test():
def testfunc(self):
print(self)
print(self.__class__)
t=Test()
t.testfunc()
输出为:
<__main__.Test object at 0x000002DF2860BD88>
<class '__main__.Test'>
self在类的方法中可以不写吗?
不可以,在Python的解释器内部,当我们调用t.testfunc()时,实际上Python解释成Test.testfunc(t),也就是说把self替换成类的实例。
class Test():
def testfunc():
print(self)
print(self.__class__)
t=Test()
t.testfunc()
报错:
TypeError: testfunc() takes 0 positional arguments but 1 was given
类方法除外
即在定义和调用时均不传实例
class Test():
def testfunc():
print(__class__)
Test.testfunc()#<class '__main__.Test'>
在继承时,传入的是哪个实例,self就是指那个传入的实例,而不是指定义了self的类的实例,即:self总是指调用时的类的实例。
class Test1():
def testfunc1(self):
print(self)
class Test2(Test1):
def testfunc2(self):
print(self)
cc=Test2()
cc.testfunc2()
cc.testfunc1()
tt=Test1()
tt.testfunc1()
运行cc.testfunc2()时,指的是Test2类的实例。
运行cc.testfunc1()时,等同于Test2.testfunc1(cc),所以self指的依然是Test2类的实例,由于self中没有定义testfunc1()方法,所以沿着继承树往上找,发现在父类Test1中定义了testfunc1()方法,所以就会成功调用。
所以输出为:
<__main__.Test2 object at 0x0000020210152BC8>
<__main__.Test2 object at 0x0000020210152BC8>
<__main__.Test1 object at 0x0000020210152C48>
self 参数的具体作用是什么呢?
打个比方,如果把类比作造房子的图纸,那么类实例化后的对象是真正可以住的房子。根据一张图纸(类),我们可以设计出成千上万的房子(类对象),每个房子长相都是类似的(都有相同的类变量和类方法),但它们都有各自的主人,那么如何对它们进行区分呢?
当然是通过 self 参数,它就相当于每个房子的门钥匙,可以保证每个房子的主人仅能进入自己的房子(每个类对象只能调用自己的类变量和类方法)。
也就是说,同一个类可以产生多个对象,当某个对象调用类方法时,该对象会把自身的引用作为第一个参数自动传给该方法,换句话说,Python 会自动绑定类方法的第一个参数指向调用该方法的对象。如此,Python解释器就能知道到底要操作哪个对象的方法了。
因此,程序在调用实例方法和构造方法时,不需要手动为第一个参数传值。
通过self调用类的成员变量:
class Test():
def __init__(self,name,age):
self.name=name
self.age=age
def testfunc1(self):
print("I'm",self.name,",",self.age,"years old")
te1=Test("Tom",19)
te1.testfunc1()#I'm Tom , 19 years old
te2=Test("lili",17)
te2.testfunc1()#I'm lili , 17 years old
te1在进行初始化时,调用的构造函数中 self 代表的是 “Tom 19”;而 te2 在进行初始化时,调用的构造函数中 self 代表的是"lili 17"
通过self调用类的方法:
class Test():
def __init__(self,number1,number2):
self.number1=number1
self.number2=number2
def add(self):
return self.number2+self.number1
def multip(self):
return self.number2 * self.number1
def show(self):
print("Result Add",self.add())
print("Result Mul:",self.multip())
te=Test(12,5)
te.show()
#Result Add 17
#Result Mul: 60
1、init 方法的第一个参数永远是 self ,表示创建的实例本身,因此,在 init 方法的内部,就可以把各种属性绑定到 self,因为 self 就指向创建的 实例本身
2、使用了 init 方法,在创建实例的时候就不能传入 空的参数了,必须传入与 init 方法匹配的参数,但是 self 不需要传,python解释器会自己把实例变量传进去