继承
概念
现实中的继承 在现实生活中,继承一般指的是子女继承父辈的财产
程序中的继承 在程序中,继承描述的是多个类之间的所属关系。
单继承
单继承:子类只继承一个父类
说明: 虽然子类没有定义__init__方法初始化属性,也没有定义实例方法,但是父类有。所以只要创建子类的对象,就默认执行了那个继承过来的__init__方法
总结:
子类在继承的时候,在定义类时,小括号()中为父类的名字
父类的属性、方法,会被继承给子类
多继承
多继承:子类继承多个父类
说明: 多继承可以继承多个父类,也继承了所有父类的属性和方法 注意:如果多个父类中有同名的 属性和方法,则默认使用第一个父类的属性和方法(根据类的魔法属性mro的顺序来查找) 多个父类中,不重名的属性和方法,不会有任何影响。
子类重写父类的同名属性和方法
例子:
输出结果:
子类调用父类同名属性和方法
例子:
输出结果:
多层继承
例子:
调用父类方法super()
私有权限
封装的意义: 将属性和方法放到一起做为一个整体,然后通过实例化对象来处理; 隐藏内部实现细节,只需要和对象及其属性和方法交互就可以了; 对类的属性和方法增加 访问权限控制。
私有权限:在属性名和方法名 前面 加上两个下划线 __ 类的私有属性 和 私有方法,都不能通过对象直接访问,但是可以在本类内部访问; 类的私有属性 和 私有方法,都不会被子类继承,子类也无法访问; 私有属性 和 私有方法 往往用来处理类的内部事情,不通过对象处理,起到安全作用。
私有属性和私有方法
例子:
类部调用私有属性和私有方法
例子:
子类不能继承父类私有属性和方法
例子:
私有属性和私有方法
总结:
私有属性,可以在类内部通过self调用,但不能通过对象访问
私有方法,可以在类内部通过self调用,但不能通过对象访问
对象不能访问私有权限的属性和方法
子类不能继承父类私有权限的属性和方法
Python中没有像C++中 public 和 private, protected 这些关键字来区别公有属性和私有属性。 Python是以属性命名方式来区分,如果在属性和方法名前面加了2个下划线'__',则表明该属性和方法是私有权限,否则为公有权限。
修改私有属性的值
修改私有属性的值
总结: 现代软件开发中,通常会定义get_xxx()方法和set_xxx()方法来获取和修改私有属性值 get_xxx()方法-->返回私有属性的值 set_xxx()方法-->接收参数,修改私有属性的值 对象不能访问私有权限的属性和方法,可以通过访问公有方法set_money()来修改私有属性的值,可以通过访问公有方法get_money()来获取私有属性的值
类属性和实例属性
类属性和实例属性
在前面的例子中我们接触到的就是实例属性(实例对象属性),顾名思义,类属性就是类对象所拥有的属性,它被所有类对象的实例对象所共有,在内存中只存在一个副本,这个和C++中类的静态成员变量有点类似。对于公有的类属性,可以通过类或者实例对象访问
实例属性(对象属性)
通过实例(对象)不能修改类属性
静态方法和类方法
类方法
类方法是类对象所拥有的方法,需要用修饰器@classmethod来标识其为类方法,对于类方法,第一个参数必须是类对象,一般以cls作为第一个参数(当然可以用其他名称的变量作为其第一个参数,但是大部分人都习惯以‘cls’作为第一个参数的名字,就最好用‘cls’了),能够通过实例对象和类名去访问。
静态方法
静态方法中不需要额外定义参数,调用静态方法可以通过对象或者类调用
实例方法
实例方法中需要self参数,因此调用实例方法需要通过实例对象调用
总结: 类方法使用@classmethod装饰,第一个参数为类(cls),调用时可以通过类的实例或者类本身来调用。 静态方法使用@staticmethod装饰,调用时可以使用类的实例或者类本身来调用。 实例方法定义时第一个参数为类的一个实例(self),调用时必须通过实例调用。
__new__方法
总结
__new__至少要有一个参数cls,代表要实例化的类,此参数在实例化时由Python解释器自动提供
__new__必须要有返回值,返回实例化出来的实例,这点在自己实现__new__时要特别注意,可以return父类__new__出来的实例,或者直接是object的__new__出来的实例 3). __init__有一个参数self,就是这个__new__返回的实例,__init__在__new__的基础上可以完成一些其它初始化的动作,__init__不需要返回值
我们可以将类比作制造商,__new__方法就是前期的原材料购买环节,__init__方法就是在有原材料的基础上,加工,初始化商品环节
单例模式
单例模式:永远用一个对象得实例,避免新建太多实例浪费资源
实质:使用__new__方法新建类对象时先判断是否已经建立过,如果建过就使用已有的对象