类与对象
把一组数据结构和处理它们的方法组成对象(object),把相同行为的对象归纳为类(class),通过类的封装(encapsulation)隐藏内部细节,通过继承(inheritance)实现类的特化(specialization)和泛化(generalization),通过多态(polymorphism)实现基于对象类型的动态分派。
简单来说,类是对象的蓝图和模板,而对象是类的实例。类是抽象的概念,而对象是具体的东西。在面向对象编程的世界,一切皆为对象,对象都有属性和行为,每个对象都是独一无二的,而且对象一定属于某个类(型)。当我们把一大堆拥有共同特征的对象的静态特征(属性)和动态特征(行为)都抽取出来后,就可以定义出“类”。
类
- 定义类
在Python中可以使用‘class’关键字定义类,然后在类中通过函数来定义方法,这样可以将对象的动态特征描述出来。
class Mask(object):
# __init__是一个特殊方法用于在创建对象时进行初始化操作
# 通过这个方法我们可以为口罩对象绑定种类和过滤水平两个属性
def __init__(self, kind, level):
self.kind = kind
self.level = level
def kind(self, name):
print('%s属于%s.' % (name, self.kind))
# PEP 8要求标识符的名字用全小写多个单词用下划线连接
# 但是部分程序员和公司更倾向于使用驼峰命名法(驼峰标识)
def mask_level(self):
if self.level > 99:
print('%s已经达到最高过滤水平.' % self.kind)
- 创建和使用对象
定义好一个类,就可以创建对象并给对象发消息。
def main():
# 创建口罩对象并指定种类和过滤水平
mask1 = Mask('N', 95)
# 给对象发kind消息
mask1.kind('N95口罩')
# 给对象发mask_level消息
mask1.mask_level()
mask2 = Mask('R', 99.97)
stu2.kind('3M P')
stu2.mask_level()
if __name__ == '__main__':
main()
访问可见性问题
在Python中,属性和方法的访问权限只有两种,也就是公开的和私有的,如果希望属性是私有的,在给属性命名时可以用两个下划线作为开头。但是,Python并没有从语法上严格保证私有属性或方法的私密性,它只是给私有的属性和方法换了一个名字来妨碍对它们的访问,事实上如果知道更换名字的规则仍然可以访问它们。在实际开发中,并不建议将属性设置为私有的,因为这会导致子类无法访问。
class Test:
def __init__(self, foo):
self.__foo = foo
def __bar(self):
print(self.__foo)
print('__bar')
def main():
test1 = Test('hello')
# AttributeError: 'Test' object has no attribute '__bar'
test1.__bar()
# AttributeError: 'Test' object has no attribute '__foo'
print(test1.__foo)
test2 = Test('hello')
# hello __bar
test2._Test__bar()
# __bar
print(test2._Test__foo)
if __name__ == "__main__":
main()
子类重写父类方法前先调用父类的方法
1.调用未绑定的父类方法
2.调用super()函数
关于类的BIF
关于属性的一些操作
利用属性来定义属性(有利于后期程序修改时,减少修改量)