面向对象编程(Object Oriented Programming,OOP,面向对象程序设计)
面向对象编程的2个非常重要的概念: 类和对象
对象是面向对象编程的核心,在使用对象的过程中,为了将具有共同特征和行为的一组对象抽象定义,提出了另外一个新的概念——类 类就相当于制造飞机时的图纸,用它来进行创建的飞机就相当于对象
1、类
人以类聚 物以群分。
具有相似内部状态和运动规律的实体的集合(或统称为抽象)。
具有相同属性和行为事物的统称
类是抽象的,在使用的时候通常会找到这个类的一个具体的存在,使用这个具体的存在。
一个类可以找到多个对象
2. 对象
某一个具体事物的存在 ,在现实世界中可以是看得见摸得着的。
可以是直接使用的
类的构成
类(Class) 由3个部分构成
01). 类的名称:类名
02). 类的属性:一组数据
03). 类的方法:允许对进行操作的方法 (行为)
例:
定义一个类,格式如下:
class 类名:
方法列表
类名的命名规则按照"大驼峰命名法"
创建对象
python中,可以根据已经定义的类去创建出一个或多个对象。
创建对象的格式为:(实例化对象)
对象名1 = 类名()
对象名2 = 类名()
对象名3 = 类名()
例:
在方法内通过self获取对象属性
魔法方法__init__
Python 的类里提供的,两个下划线开始,两个下划线结束的方法,就是魔法方法,__init__()就是一个魔法方法,通常用来做属性初始化 或 赋值 操作(作用)。
__init__()方法,在创建一个对象时默认被调用,不需要手动调用
__init__(self)中的self参数,不需要开发者传递,python解释器会自动把当前的对象引用传递过去。
有参数的__init__()方法
有参数的__init__(self)方法
说明:
通过一个类,可以创建多个对象,就好比 通过一个模具创建多个实体一样
__init__(self)中,默认有1个参数名字为self,如果在创建对象时传递了2个实参,那么__init__(self)中出了self作为第一个形参外还需要2个形参,例如__init__(self,x,y)
注意: 1). 在类内部获取 属性 和 实例方法,通过self获取;
2). 在类外部获取 属性 和 实例方法,通过对象名获取。
3). 如果一个类有多个对象,每个对象的属性是各自保存的,都有各自独立的地址;
4). 但是实例方法是所有对象共享的,只占用一份内存空间。类会通过self来判断是哪个对象调用了实例方法。
魔法方法:__str__()方法
# 查看类的文档说明,也就是类的注释
# print(Hero.__doc__)
# print(Hero.属性名.__doc__)
说明: 在python中方法名如果是__xxxx__()的,那么就有特殊的功能,因此叫做“魔法”方法 当使用print输出对象的时候,默认打印对象的内存地址。如果类定义了__str__(self)方法,那么就会打印从在这个方法中 return 的数据
__str__方法通常返回一个字符串,作为这个对象的描述信息
魔法方法:__del__()方法
说明:
1). 创建对象后,python解释器默认调用__init__()方法;
2). 当删除对象时,python解释器也会默认调用一个方法,这个方法为__del__()方法
例:
总结:
1). 当有变量保存了一个对象的引用时,此对象的引用计数就会加1;
2). 当使用del() 删除变量指向的对象时,则会减少对象的引用计数。如果对象的引用计数不为1,那么会让这个对象的引用计数减1,当对象的引用计数为0的时候,则对象才会被真正删除(内存被回收)。
练习:
创建一个狗类:(Dog)
创建一个大黄狗,5岁,技能“汪汪叫”
创建一个大黑狗,3岁,技能“龇牙咧嘴”
创建一个大白狗,1岁,技能“夹着尾巴跑” 并输出
继承介绍以及单继承
在程序中,继承描述的是多个类之间的所属关系。
如果一个类A里面的属性和方法可以复用,则可以通过继承的方式,传递到类B里。 那么类A就是基类,也叫做父类;类B就是派生类,也叫做子类。
例:
单继承:子类只继承一个父类
说明: 虽然子类没有定义__init__方法初始化属性,也没有定义实例方法,但是父类有。所以只要创建子类的对象,就默认执行了那个继承过来的__init__方法
总结:
子类在继承的时候,在定义类时,小括号()中为父类的名字
父类的属性、方法,会被继承给子类
例:
多继承:子类继承多个父类
说明: 多继承可以继承多个父类,也继承了所有父类的属性和方法
注意:如果多个父类中有同名的 属性和方法,则默认使用第一个父类的属性和方法(根据类的魔法属性mro的顺序来查找)
多个父类中,不重名的属性和方法,不会有任何影响。
例1:
例2:
子类重写父类的同名属性和方法
多层继承
调用父类方法super()