Python 中的继承、多态和封装
涉及问题:
Python 中如何实现多继承,会有什么问题?
Python 中的多态与静态方法有什么区别?
答案要点如下:
- Python 中的继承,就是在定义类时,在括号中声明父类,简单示例如下:
class Father(object): # object 是最基础的一个类,和 JAVA 中的 Object 是一样的
pass
class Chile(Father): # 继承 Father 类
pass
- 我们都知道,在定义类时,可以通过定义 __init__ 方法来初始化类的属性。有点类似于 JAVA 中的有参构造。但不同的是,在 JAVA 中,子类的构造函数会默认调用父类的无参构造,即在构造函数中的无论你写与不写,第一句代码都是 super() 。而在 Python 中不是的,如果子类不重写 __init__ 方法,默认会调用父类的 __init__ 。而一旦子类自己定义了 __init__,则不再会调用父类中的方法,如果想调用,需要手动通过 super() 来调用。示例代码如下:
class Father(object):
f_out = "123"
def __init__(self, f):
print("father class")
self.f = f
class Child_1(Father):
pass
class Child_2(Father):
# 如果子类中不重写 __init__ 方法,会默认调用父类中的 __init__ 方法
# 但是如果重写后,则不会自动调用父类中的 __init__ 方法,需要手动来调用
def __init__(self):
# 如果子类在初始化参数时,没有父类的参数,则子类不再有父类拥有的实例属性
# 但类属性仍然会被继承
print("child class")
print(self.f_out)
# 如果想调用父类
# super(Child_2, self).__init__("123")
# c1 = Child_1() # 如果不传参会报错说接受一个参数,这里就已经说明在调用父类的 __init__
c1 = Child_1("123") # 输出 father class,且必须传入一个参数
print(c1.f) # 得到 123
c2 = Child_2() # 不接受参数,因为子类中没有参数
# 输出 child class 123
# print(c2.f) # 报错,找不到 attribute f
- super() 方法的说明,第一个参数是当前类,即子类,第二个参数固定 self,表示当前的实例对象,在 Python2 中必须加参数调用,而 python3 中可以省略参数,如下所示:
# python2
super(Child, self).父类方法(*args, **kw)
# python3
super().父类方法(*args, **kw)
- Python 相比于 JAVA 更好的一点在于支持多继承,而 JAVA 是单一继承的。在 JAVA 中实现多继承可以通过接口与内部类(这也是我曾经遇到过的一个面试题,有兴趣的可以自行查找资料)。在 Python 实现多继承就很简单了,示例如下:
class Father():
pass
class Mother():
pass
class Child(Father, Mother):
pass
- 需要注意的是,在 Python 中多继承的调用会存在一些问题,有时候会出现我们意想不到的结果,这里不详细展开,感兴趣的可以自己了解,后面会再开文章进行介绍。主要示例代码如下:
class Base(object):
def __init__(self):
print("enter base")
print('leave base')
class A(Base):
def __init__(self):
print('enter A')
super(A, self).__init__()
print('leave A')
class B(Base):
def __init__(self):
print('enter B')
super(B, self).__init__()
print('leave B')
class C(A,B):
def __init__(self):
print('enter C')
super(C, self).__init__()
print('leave C')
c = C()
# 输出结果如下
‘’‘
enter C
enter A
enter B
enter base
leave base
leave B
leave A
leave C
’‘’
- Python 中的封装。主要是对属性的封装,采用 __属性名 的形式,(注意是两个下划线)。在 Python 中,以两个下划线开头和结尾,是 Python 中的一些特殊变量,所以我们在私有化属性时,一般不这样定义。而以一个下划线开头的属性,可以通过 实例名. 的方式进行调用,但它有个约定俗成的含义,即:我可以通过 实例. 来调用,但请把我视为 私有变量。(这也很符合 Python 中体现的一切靠自觉的思想)。示例如下:
class Test(object):
def __init__(self, name, age, sex):
self.__name = name
self._age = age
self.sex = sex
t = Test("Demon", 18, 'M')
# print(t.__name) # 'Test' object has no attribute '__name'
print(t._Test__name) # 可以这样访问到,但不要这样做
print(t._age) # 18
print(t.sex) # M
- Python 中的多态。在 JAVA 中,用一句话总结多态就是 父类引用指向子类对象。而在 Python 中,父类引用指向子类对象也是多态的一种实现,但不同的是 Python 中多了一种鸭子类型,即如果一个动物,叫起来像鸭子,走起来像鸭子,跑起来像鸭子,那我们就认为它是一只鸭子。即如果一个类,它有和别的类相同的方法,我们就认为它和这个类具有某种关系,是类似的一种类。示例如下:
class Animal(object):
def run(self):
print('Animal run')
# 定义一个 Animal 的子类
class Dog(Animal):
def run(self):
print('Dog run')
# 定义一个类似 Animal 的类
class Like_Animal(object):
# 同样具有 run() 方法
def run(self):
print('I can run too!!!')
# 定义一个方法,参数是实例对象 ,根据参数来调用对应的 run() 方法
# 也就是我们说的多态
def start_run(run_obj):
run_obj.run()
# 传入一个 Animal 实例对象
start_run(Animal())
# 传入一个 Dog 实例对象
start_run(Dog())
# 传入一个 类Animal 实例对象
start_run(Like_Animal())
扩展:
- 关于多继承,涉及到 Python 中的 MRO ,后面会再详细介绍
- 关于封装,还可以使用 @property ,这个也比较常用,可以自己了解下
- http://www.aibbt.com/a/22236.html
Python 中的继承、多态和封装的更多相关文章
- python面向对象之继承/多态/封装
老师说,按继承/多态/封装这个顺序来讲. 子类使用父类的方法: #!/usr/bin/env python # coding:utf-8 class Vehicle: def __init__(sel ...
- python中的继承和多态
继承 继承的表现方式: class Animal(): pass class Cat(Animal): #animal是cat的父类,也可以说是基类 pass print(Cat.__bases__) ...
- python中的继承原则
继承是面向对象的重要特征之一,继承是两个类或者多个类之间的父子关系,子进程继承了父进程的所有公有实例变量和方法.继承实现了代码的重用.重用已经存在的数据和行为,减少代码的重新编写,python在类名 ...
- Python中的继承
继承: 面向对象程序语言的一个重要特点是继承.继承提供了在已存在类的基础上创建新类的方法.继承的子类 拥有被继承的父类的所有方法,在此基础上,子类还可以添加自己的专有方法.继承是类的强有力的特点.一些 ...
- python中多继承C3算法研究
在python的面向对象继承问题中,单继承简单易懂,全部接受传承类的属性,并可添加自带属性, 但是,在多继承情况下,会遇到多个被继承者的顺序问题,以及多次继承后查找前几次继承者需求属性时,可能不易发现 ...
- Python之浅谈多态和封装
目录 组合 什么是组合 为什么使用组合 多态和多态性 多态 什么是多态? 多态性 好处 多态性 什么是多态性 封装 封装是什么意思? 隐藏 如何用代码实现隐藏 python 实际上是可以访问隐藏属性的 ...
- python中 的继承
1.Python的类可以继承多个类,Java和C#中则只能继承一个类. 2.Python的类如果继承了多个类,那么其寻找方法的方式有两种,分别是:深度优先和广度优先. 当类是经典类时,多继承情况下,会 ...
- Python 中 对logging 模块进行封装,记录bug日志、日志等级
是程序产生的日志 程序员自定义设置的 收集器和渠道级别那个高就以那个级别输出 日志和报告的作用: 报告的重点在于执行结果(执行成功失败,多少用例覆盖),返回结果 日志的重点在执行过程当中,异常点,哪里 ...
- 理解Python中的继承规则和继承顺序
先来看一段代码: class First(object): def __init__(self): print ("first") class Second(object): de ...
随机推荐
- LeetCode第四天
leetcode 第四天 2018年1月4日 15.(628)Maximum Product of Three Numbers JAVA class Solution { public int max ...
- hihoCoder Demo Day dp
题意:有一个机器人被困在一个的迷宫中,机器人的初始位置是,目的地是,并且它的移动方式很奇怪:只能一直向右,直到不能再向右才能把方向变成向下:只能一直向下,直到不能再向下才能把方向变成向右.迷宫中的每个 ...
- Minikube之Win10单机部署
Kubernetes(k8s)是自动化容器操作的开源平台,基于这个平台,你可以进行容器部署,资源调度和集群扩容等操作.如果你曾经用过Docker部署容器,那么可以将Docker看成Kubernetes ...
- (转载)SVM-基础(一)
支持向量机: Maximum Margin Classifier by pluskid, on 2010-09-08, in Machine Learning 87 comments 本文是 ...
- CDN中前端层的复制
前端层的复制是为了提高静态内容分发的性能和可扩展性.将静态内容的分发转由边缘服务器来完成是为了解决可扩展性的问题,因为这样做可以避免出现在对等点和广域网链路处的网络拥堵风险,而这两个地方的拥堵是网络延 ...
- 关于 target="_blank"漏洞的分析
创建: 于 八月 30, 2016 关于 target="_blank"漏洞的分析 一.漏洞详情:首先攻击者能够将链接(指向攻击者自己控制的页面的,该被控页面的js脚本可以对母页 ...
- python︱模块加载(pip安装)以及pycharm安装与报错解决方式
每每以为攀得众山小,可.每每又切实来到起点,大牛们,缓缓脚步来俺笔记葩分享一下吧,please~ --------------------------- 准备放下R开始学python,真是痛苦,因为找 ...
- java Socket实现简单在线聊天(二)
接<java Socket实现简单在线聊天(一)>,在单客户端连接的基础上,这里第二步需要实现多客户端的连接,也就需要使用到线程.每当有一个新的客户端连接上来,服务端便需要新启动一个线程进 ...
- Flex父子窗口相互调用
Flex父子窗口相互调用 1.设计思路 (1)子窗口调用父窗口的方法 (2)子窗口做了修改后,返回父窗口,父窗口调用子窗口函数 2.设计源码 (1)父窗口 ParentWindow.mxml: < ...
- 修改MyEclipse行数的颜色
修改MyEclipse行数的颜色 1.未修改前,行数的颜色 2.依次选择"Window--->Preferences" 3.选择"General--->Edi ...