常见概念一览
概念 | 解释 |
---|---|
类(Class) | 用来描述具有相同的属性和方法的对象的集合。它定义了该集合中每个对象所共有的属性和方法。对象是类的实例 |
类变量 | 类变量在整个实例化的对象中是公用的。类变量定义在类中且在函数体之外。类变量通常不作为实例变量使用 |
数据成员 | 类变量或者实例变量, 用于处理类及其实例对象的相关的数据 |
方法重写 | 如果从父类继承的方法不能满足子类的需求,可以对其进行改写,这个过程叫方法的覆盖(override),也称为方法的重写 |
局部变量 | 定义在方法中的变量,只作用于当前实例的类 |
实例变量 | 在类的声明中,属性是用变量来表示的。这种变量就称为实例变量,是在类声明的内部但是在类的其他成员方法之外声明的 |
继承 | 即一个派生类(derived class)继承基类(base class)的字段和方法。继承也允许把一个派生类的对象作为一个基类对象对待。例如,有这样一个设计:一个Dog类型的对象派生自Animal类,这是模拟"是一个(is-a)"关系(例图,Dog是一个Animal) |
实例化 | 创建一个类的实例,类的具体对象 |
方法 | 类中定义的函数 |
对象 | 通过类定义的数据结构实例。对象包括两个数据成员(类变量和实例变量)和方法 |
定义一个类
在Python中可以使用class
关键字定义类,然后在类中通过之前学习过的函数来定义方法,这样就可以将对象的动态特征描述出来,代码如下所示。
class Student(object):
# __init__是一个特殊方法用于在创建对象时进行初始化操作
# 通过这个方法我们可以为学生对象绑定name和age两个属性
def __init__(self, name, age):
self.name = name
self.age = age
def study(self, course_name):
print('%s正在学习%s.' % (self.name, course_name))
# PEP 8要求标识符的名字用全小写多个单词用下划线连接
# 但是部分程序员和公司更倾向于使用驼峰命名法(驼峰标识)
def watch_movie(self):
if self.age < 18:
print('%s只能观看《熊出没》.' % self.name)
else:
print('%s正在观看岛国爱情大电影.' % self.name)
说明: 写在类中的函数,我们通常称之为(对象的)方法,这些方法就是对象可以接收的消息。
创建和使用对象
当我们定义好一个类之后,可以通过下面的方式来创建对象并给对象发消息。
def main():
# 创建学生对象并指定姓名和年龄
stu1 = Student('小明', 18)
# 给对象发study消息
stu1.study('Python程序设计')
# 给对象发watch_av消息
stu1.watch_movie()
stu2 = Student('王大锤', 15)
stu2.study('思想品德')
stu2.watch_movie()
if __name__ == '__main__':
main()
私有和公有属性
在Python中的所有属性都是public,可能有c++和java的同学觉得神奇,其实python最初规定了一种特殊的命名方式来区分public还是private,那就是在给属性命名时可以用两个下划线作为开头。
class Car:
wheels = 0
def __init__(self, color, model, year):
self.color = color
self.model = model
self.year = year
self.__cupholders = 6
my_car = Car("yellow", "Beetle", "1969")
print(f"It was built in {my_car.year}")
print(f"It has {my_car.__cupholders} cupholders.")
Out:
It was built in 1969
---------------------------------------------------------------------------
AttributeError Traceback (most recent call last)
<ipython-input-108-1efe56f0c054> in <module>
1 my_car = Car("yellow", "Beetle", "1969")
2 print(f"It was built in {my_car.year}")
----> 3 print(f"It has {my_car.__cupholders} cupholders.")
AttributeError: 'Car' object has no attribute '__cupholders'
但是,Python并没有从语法上严格保证私有属性或方法的私密性,它只是给私有的属性和方法换了一个名字来妨碍对它们的访问,事实上如果你知道更换名字的规则仍然可以访问到它们。
当Python看到__ 时,会自动在cupholders前面补上一个下划线_和所属类名,也就是说,这里我们尝试用my_car.__cupholders 来调用时,Python默认的正确写法是
my_car._Car__cupholders,现在再试一下:
print(f"It has {my_car._Car__cupholders} cupholders")
Out: It has 6 cupholders
访问权限管理
就像刚刚提到的,Python所有的东西都是公有的,我们可以随意的新增,修改,甚至删除变量:
my_car = Car("yellow", "beetle", 1969)
print(f"My car was built in {my_car.year}")
my_car.year = 2003
print(f"It was built in {my_car.year}")
del my_car.year
print(f"It was built in {my_car.year}")
Out:
My car was built in 1969
It was built in 2003
---------------------------------------------------------------------------
AttributeError Traceback (most recent call last)
<ipython-input-110-46914b0bae82> in <module>
6
7 del my_car.year
----> 8 print(f"It was built in {my_car.year}")
AttributeError: 'Car' object has no attribute 'year'
那我们如何才能控制属性的访问权限呢?Python给出的答案是装饰器 @property。
class Car:
def __init__(self, color, model, year):
self.color = color
self.model = model
self.year = year
self._voltage = 12
@property
def voltage(self):
return self._voltage
@voltage.setter
def voltage(self, volts):
print("Warning: this can cause problems!")
self._voltage = volts
@voltage.deleter
def voltage(self):
print("Warning: the radio will stop working!")
del self._voltage
我们新增了voltage(电压)这个属性,并用property来控制外部的访问权限,这里我们定义了三个方法,利用setter方法可以改变voltage的值,利用getter方法来访问,利用deleter方法实现删除,接下来让我们新建实例来看看propert是如何工作的:
my_car = Car("yellow", "beetle", 1969)
print(f"My car uses {my_car.voltage} volts")
my_car.voltage = 6
print(f"My car now uses {my_car.voltage} volts")
del my_car.voltage
Out:
My car uses 12 volts
Warning: this can cause problems!
My car now uses 6 volts
Warning: the radio will stop working!
可以发现,我们这里直接使用.voltage 而不是._voltage,这样就告诉python去使用property装饰的方法,我们可以通过使用@.setter and @.deleter 使属性变为read-only(只读),从而保护voltage不会被随意修改和删除