大家都知道面向对象的三大特性:封装,继承,多态。封装特性在上一章节已经讲解过。这一章节主要讲解继承和多态。

继承:

当定义一个类的时候,可以从现有的类进行继承。那么新定义的类可以称为子类,被继承的现有的类称为基类,父类或者超类。

上一章节大家应该有所注意以下代码:

>>> class Animal(object):
pass

object类是python所有类的基类,Animal类是object类的子类。Animal类继承于object类。

继承到底有什么好处?举个例子:

>>> class Animal(object):
def call(self):
print('Animal is calling!')
>>> class dog(Animal):
pass
>>> dog = dog()
>>> dog.call()
Animal is calling!

dog类继承了Animal类,当声明了一个dog实例对象后,可以直接调用父类Animal的call方法。

这就是继承的好处,能够直接继承来自父类的属性和方法,然后直接调用。这样就精简了代码,提高了代码的复用性。

如果父类在实例化对象的时候,要默认传入name参数的话,子类dog继承的时,应该怎么办?

例子:

>>> class Animal(object):
def __init__(self,name):
self.name = name
def call(self):
print ('Animal is calling!') >>> class dog(Animal):
def __init__(self,name,birth):
super(dog,self).__init__(name)//大家注意这句代码
self.birth = birth
def call(self):
print ('dog is calling!') >>> dog = dog('wangcai','2016-8-5')
>>> print dog.name
wangcai
>>> print dog.birth
2016-8-5

函数super(dog, self)将返回当前类继承的父类,即Animal,然后调用__init__()方法,注意self参数已在super()中传入,在__init__()中将隐式传递,不需要写出(也不能写).

大家发现,dog子类也定义了一个call方法。那么实例化dog对象,dog.call()会发生什么?

多态

>>> dog.call()
dog is calling!

当子类和父类都存在相同的call()方法时,我们说,子类的call()覆盖了父类的call(),在代码运行的时候,总是会调用子类的call()。类似于C++的虚函数或者java的抽象方法。

在理解多态之前,要判断一个对象/变量是否属于某一个类/类型,使用isinstance方法。

>>> class Animal(object):
def call(self):
print('Animal is calling!') >>> class dog(Animal):
def call(self):
print('dog is calling!') >>> d =dog()
>>> a =Animal()
>>> isinstance(d,Animal)
True
>>> isinstance(a,dog)
False

看来,d不仅仅是dog,而且还是animal。而a 虽然是animal,但是不一定是dog,也有可能是cat。所以isinstance(a,dog)返回为false。

所以在继承关系中,如果一个实例的数据类型是某个子类,那么它的数据类型也可以被看做是父类。所以isinstance(d,Animal)返回为True。

有个上述的理解,那么我们再举个例子。

>>> def lound_call(animal):
animal.call()
>>> lound_call(dog())
dog is calling!
>>> lound_call(Animal())
Animal is calling!

新增一个Animal子类cat。

>>> class Cat(Animal):
def call(self):
print('cat is calling!')
>>> lound_call(Cat())
cat is calling!

你会发现,新增一个Animal的子类,不必对lound_call()做任何修改,实际上,任何依赖Animal作为参数的函数或者方法都可以不加修改地正常运行,原因就在于多态。

多态的好处就是,当我们需要传入DogCat……时,我们只需要接收Animal类型就可以了,因为DogCat……都是Animal类型,然后,按照Animal类型进行操作即可。由于Animal类型有call()方法,因此,传入的任意类型,只要是Animal类或者子类,就会自动调用实际类型的call()方法,这就是多态的意思:

对于一个变量,只需要知道它是Animal类型,无需确切地知道它的子类型,就可以放心地调用call()方法,而具体调用的call()方法是作用在AnimalDogCat对象上,由运行时该对象的确切类型决定,这就是多态真正的威力:调用方只管调用,不管细节,而当我们新增一种Animal的子类时,只要确保call()方法编写正确,不用管原来的代码是如何调用的。这就是著名的“开闭”原则:

对扩展开放:允许新增Animal子类;

对修改封闭:不需要修改依赖Animal类型的lound_call()等函数。

小结:这一章节主要讲解了python面向对象的属性继承和多态。

python学习第十五天 -面向对象之继承和多态的更多相关文章

  1. Py修行路 python基础 (十五)面向对象编程 继承 组合 接口和抽象类

    一.前提回忆: 1.类是用来描述某一类的事物,类的对象就是这一类事物中的一个个体.是事物就要有属性,属性分为 1:数据属性:就是变量 2:函数属性:就是函数,在面向对象里通常称为方法 注意:类和对象均 ...

  2. Python学习(十二) —— 面向对象

    一.初识面向对象 面向过程的核心是过程,流水线思维,过程即解决问题的步骤,面向过程的设计就好比精心设计好一条流水线,考虑周全什么时候处理什么东西. 优点:极大地降低了写程序的复杂度,只需要顺着要执行的 ...

  3. JavaSE学习总结(五)——封装,继承,多态很简单

    java面向对象的三大特性是:封装.继承与多态,是面向对象编程的核心. 一.封装 简单说封装就是将同一类事物的特性与功能包装在一起,对外暴露调用的接口. 封装:封装也称信息隐藏,是指利用抽象数据类型把 ...

  4. Python学习第十五篇——类继承和类实例化

    学习Python类时,我们明白了类的本质,以及所谓的面向对象编程思想强调的对事物本身的属性,我们对某一类事物进行描述——采用了很多方法,这些方法描述了类的属性(比如猫科动物的眼睛,四肢,是否哺乳类等等 ...

  5. python学习第十四天 -面向对象编程基础

    python也是支持面向对象编程的.这一章节主要讲一些python面向对象编程的一些基础. 什么是面向对象的编程? 1.面向对象编程是一种程序设计范式 2.把程序看做不同对象的相互调用 3.对现实世界 ...

  6. Python 学习 第十五篇:模块搜索路径和包导入

    在导入自定义的模块时,除了指定模块名之外,也需要指定目录,由于Python把目录称作包,因此,这类导入被称为包导入.包导入把计算机上的目录变成Python的命名空间,而目录中所包含的子目录和模块文件则 ...

  7. Python学习札记(十五) 高级特性1 切片

    参考: 高级特性 切片 Note 1.掌握了Python的基础语法之后,就可以写出很多很有用的程序了,比如打印1-90的奇数: #!/usr/bin/env python3 L = [] n = 1 ...

  8. python学习(十四)面向对象

    Python中的面向对象,先写类,会生成类对象,类对象然后创建对象,对象就可以拿来用了. Python支持多重继承. class语句创建类对象,并将其赋值给变量名. class语句内的赋值语句会创建类 ...

  9. Python学习第十四课——面向对象基本思想part1

    面向对象的基本思想 # 写法1 person1 = { 'name': 'hanhan', ', 'sex': '男' } def xue_xi(person): print('%s在学习' % pe ...

随机推荐

  1. Verilog HDL中阻塞语句和非阻塞语句的区别

    在Verilog中有两种类型的赋值语句:阻塞赋值语句(“=”)和非阻塞赋值语句(“<=”).正确地使用这两种赋值语句对于Verilog的设计和仿真非常重要. Verilog语言中讲的阻塞赋值与非 ...

  2. akka简单示例-1

    刚刚开始接触akka,网上找了2个简单示例,并在公司运营机器上尝试,踩了一些坑,在此记录. 1. 本地hello world [torstan@sparkb5-i ~/akka_example/hel ...

  3. HADOOP在处理HIVE时权限错误的解决办法

    今天,小乔操作时发现问题: org.apache.hadoop.security.AccessControlException: Permission denied: user=root, acces ...

  4. java疯狂演义----简单java IDE工具

    file:commons package org.crazyit.editor.commons; import org.crazyit.editor.EditorFrame; import org.c ...

  5. Altium Designer 09 (Protel)总线使用方法(解决导入PCB无网络标号问题)

    弄了两天的Protel总线问题终于解决了,一开始顶层总线连接好后,导入PCB没有网络标号,也就是两个子图信号没连上.现在将正确的连接和设置方法公布如下: 1.首先画好子图的总线,如下图所示.注意:中括 ...

  6. 设计模式(十四):Command命令模式 -- 行为型模式

    1.概述         在软件设计中,我们经常需要向某些对象发送请求,但是并不知道请求的接收者是谁,也不知道被请求的操作是哪个,我们只需在程序运行时指定具体的请求接收者即可,此时,可以使用命令模式来 ...

  7. 近 100 个 Linux 常用命令大全

    1.ls [选项] [目录名 | 列出相关目录下的所有目录和文件 -a 列出包括.a开头的隐藏文件的所有文件 -A 通-a,但不列出"."和".." -l 列出 ...

  8. java中protect属性用法总结

    测试代码: pojo类: package com.lky.h1; public class Base { private Integer id; protected String name; publ ...

  9. AngularJs学习笔记7——四大特性之模块化设计

    模块化设计 1.引用自定义模块并调用 自定义模块中,如果有一些服务.封装好笑模块,在另外一个模块中(声明的时候,在依赖列表中加入要引入的模块) var app02 = angular.module(' ...

  10. MySQL 二进制日志(Binary Log)

    同大多数关系型数据库一样,日志文件是MySQL数据库的重要组成部分. MySQL有几种不同的日志文件.通常包括错误日志文件,二进制日志,通用日志,慢查询日志,等等.这些日志能够帮助我们定位mysqld ...