推荐b站资源:一鼓作气拿下Python的拦路虎:面向对象------苑昊老师
首次学习/快速回顾两相宜
此处仅作为个人学习笔记,仅总结自己认为的重点。
理解self
以前一篇的代码为例:
python
# 一般而言类名命名建议首字母大写,函数/方法名建议小写。
# 注意:所有标识符必须以字母或下划线开头,不能以数字开头。
class Dog:
# 定义属性
legs_num = 4
has_hair = True
has_tail = True
# 定义方法
def bark(self):
print("狂吠")
def bite(self):
print("咬人")
def fetch(self):
print("叼物品")
可以注意到,在Dog类下的方法定义中都带有一个self:
def bark(self):
......
def bite(self):
......
def fetch(self ):
......
**这里的self是一个形参,且必须作为类的方法的第一个参数,它代表调用该方法的实例对象本身,在调用方法的时候不需要显式传递这个参数(不需要给实参),Python 解释器会在背后自动完成这个传递。**self作为形参的变量名属于一种规范,它也可以叫做cow、her或cat等,但是就像是给类命名默认首字母大写一样,默认使用self作为这个形参的变量名。
用以下代码来进一步说明。首先对Dog类里的bark()方法的代码稍作修改:
python
# 一般而言类名命名建议首字母大写,函数/方法名建议小写。
# 注意:所有标识符必须以字母或下划线开头,不能以数字开头。
class Dog:
# 定义属性
legs_num = 4
has_hair = True
has_tail = True
# 定义方法
def bark(self):
print(f"self is: {self}")
print("狂吠")
def bite(self):
print("咬人")
def fetch(self):
print("叼物品")
再创建alex这个实例对象,并打印alex的地址,通过alex.bark()调用实例方法,打印self的地址:
python
alex = Dog()
print(f"alex is: {alex}")
alex.bark()

从运行结果可以得出此时self就是alex这个实例对象,也就是说,当我们调用一个实例方法时,这个实例对象会作为变量自动传递给self(上面代码中调用.bark()方法的时候确实没有传递过alex这个实参),实现实例对象和方法的绑定。如果类空间中的方法被不同的实例对象调用,那么这个self则指的是不同的实例对象。
再次修改代码,给alex和peiqi分别赋予name属性,让这两个实例拥有自己的名字,这样可以更加直观的理解上面这段话的含义:
python
class Dog:
# 定义属性
legs_num = 4
has_hair = True
has_tail = True
# 定义方法
def bark(self, person):
print(f"self is: {self.name}")
print(f"{self.name}向{person}狂吠")
def bite(self):
print("咬人")
def fetch(self):
print("叼物品")
alex = Dog()
peiqi = Dog()
alex.name = "alex" # 为alex添加name属性
peiqi.name = "peiqi" # 为peiqi添加name属性
print(alex.name)
alex.bark("我")
print(peiqi.name)
peiqi.bark("你")

从运行结果看,不同的实例对象调用类空间的方法.bark()时,self指的就是当前调用该.bark()方法的实例对象本身。