在上一篇文章中,我们介绍了 Python 的函数式编程,现在我们介绍 Python 的类和继承。
查看上一篇文章请点击:https://www.cnblogs.com/dustman/p/10010690.html
类
先前,我们研究过两种编程范式--命令式(使用语句、循环和函数)和函数(使用纯函数、高阶函数和递归)。
接下来我们学习一个编程方式是面向对象编程 —— Object Oriented Programming,简称 OOP,这是一种程序设计思想。OOP 把对象作为程序的基本单元,类描述对象将是什么,一个对象包含了数据和操作数据的函数。
面向过程的程序设计把计算机程序视为一系列的命令集合,即一组函数的顺序执行为了简化程序设计,面向过程把函数继续切分为子函数,即把大块函数通过切割成小块函数来降低系统的复杂度。
而面向对象的程序设计把计算机程序视为一组对象的集合,而每个对象都可以接收其他对象发过来的消息并处理这些消息。计算机程序的执行就是一系列消息在各个对象之间传递。
类是使用关键字 class 和缩进块创建的,缩进块包含类方法(这些是函数)。
class Dog: def __init__(self,name): #__init__传参数 self.name = name d1 = Dog("张三") d2 = Dog("李四") d3 = Dog("王五")
数据封装、继承和多态是面向对象的三大特点。
__init__方法
__init__方法是类中比较重要的方法,它在创建类的实例(对象)时调用,这种方法创建的属性我们称为实例变量。
类中所有方法都必须将 self 作为它们的第一个参数,尽管它没有显式传递,但是 Python 将 self 参数添加到列表中。在调用方法时不需要包含它。在方法定义中,self 引用调用该方法的实例。类的实例具有属性,这些属性是与实例关联的数据。
在本例中,Dog 实例将具有 name 和 eyes 的属性。可以通过实例后面加点和属性名来访问这些值。同样,在 __init__ 方法中,可以使用 self.attribute 来设置实例属性的初始值。
class Dog: def __init__(self,name,eyes): #__init__传参数 self.name = name self.eyes = eyes d1 = Dog("张三",2) d2 = Dog("李四",2) d3 = Dog("王五",2) print(d1.name) print(d2.name) print(d3.name)
特殊方法 __init__ 前后分别有两个下划线! 在上面的实例中,__init__ 方法接受两个参数,并将它们分配给对象的属性。__init__ 方法称为类构造函数。
方法
类可以定义其他方法用来添加一些功能。请记住,所有方法都必须将 self 作为它们的第一个参数。使用点加属性的语法来访问这些方法。
class Dog: def __init__(self,name,eyes): #__init__传参数 self.name = name self.eyes = eyes def bulk(self): print("%s:wang wang wang!" %self.name) d1 = Dog("张三",2) d2 = Dog("李四",2) d3 = Dog("王五",2) d1.bulk() d2.bulk() d3.bulk()
运行结果:
>>> 张三:wang wang wang! 李四:wang wang wang! 王五:wang wang wang! >>>
类属性一种是通过 __init__ 方法来定义,也可以自己直接定义类属性,这种属性我们叫它类变量。它是通过在类的主体内分配变量创建的。可以从类的实例或类本身访问它们。
class Dog: def __init__(self,name,eyes): #__init__传参数 self.name = name self.eyes = eyes d1 = Dog("张三",2) d2 = Dog("李四",2) d3 = Dog("王五",2) print(d1.name) print(d2.name,d2.eyes) print(d3.eyes)
运行结果:
>>> 张三 李四 2 2 >>>
类变量由类的所有实例共享。
尝试访问一个实例中未定义的属性或方法会导致 AttributeError 异常。
class Student: def __init__(self,id,name): self.id = id self.name = name man = Student(10086,"China") print(man.id) print(man.sex)
运行结果:
>>> 10086 AttributeError: ‘Student‘ object has no attribute ‘sex‘ >>>
继承
通过在两个类中共享函数实现继承。
想像以下有些类,比如 Cat, Dog, Rabbit。尽管它们有一些不同,但是它们都有颜色,名字属性。
这些相同点可以通过继承父类 Animal 来实现这些共享的属性方法。继承最大的好处是子类获得了父类的全部功能。
class Animal: def __init__(self,name,sex): self.name = name self.sex = sex class Cat(Animal): def talk(self): print(‘Neow!‘) func = Cat("ZS","F") print(func.name) func.talk()
运行结果:
>>> ZS Neow! >>>
在 OOP 程序设计中,当我们定义了一个 class 的时候,可以从某个现有的 class 继承,新的 class 称为子类 (Subclass) , 而被继承的 class 称为基类、父类或超类 (Base class、Super class)。
如果一个子类拥有一个和父类相同的属性和方法,我们称为重写 (override)。在代码运行的时候,总是会调用子类的方法。
class Animal: def __init__(self,name,sex): self.name = name self.sex = sex def take(self): print("Hello...") class Cat(Animal): def talk(self): print(‘Neow!‘) func = Cat("ZS","F") print(func.name) func.talk()
运行结果:
>>> ZS Neow! >>>
上面例子中 Animal 是父类,Cat 是子类。
子类同样可以做为父类被继承,这样继承的子类拥有两个父类的方法和属性。
class A: def func_A(self): print("A 类") class B(A): def func_B(self): print("B 类") class C(B): def func_C(self): print("C 类") obj = C() obj.func_A() obj.func_B() obj.func_C()
运行结果:
>>> A 类 B 类 C 类 >>>
注意:尽量不要循环继承。
方法 super 用来在子类中代指父类,可以用于在实例的父类中找到具有特定名称的方法。
class Animal: def msg(self): print("It‘s a dog!") class Cat(Animal): def talk(self): print(‘Neow!‘) super().msg() Cat().msg()
运行结果:
>>> It‘s a dog! >>>
Cat().msg() 调用父类的 msg 方法。
“你们纪念的只是过去,如果拉上你的各种同学到你面前,你们还是无话可说。”
原文地址:https://www.cnblogs.com/dustman/p/10016359.html