热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Python进阶_OOP面向对象编程_组合与继承

#目录前言组合派生通过继承来覆盖重载方法最常用的重载场景实例方法的重载从标准类中派生类方法的重载前言我们定义一个类是希望能够把类当成模块来使用,并把类嵌入到我们的应用代码中,与其他的数
#目录

  • 前言
  • 组合
  • 派生
    • 通过继承来覆盖重载方法
    • 最常用的重载场景实例方法的重载
    • 从标准类中派生类方法的重载

前言

我们定义一个类是希望能够把类当成模块来使用,并把类嵌入到我们的应用代码中,与其他的数据类型、逻辑执行流结合使用。一般来说我们可以使用两种方法在代码中利用类,那就是组合和派生。

组合

组合: 就是将不同的类混合并加入到其他类中,来 增加类的功能 / 提高代码的重用性 / 易于维护(对类的修改会直接反应到整个应用中) 。我们可以实例化一个更大的对象,同时还可以添加一些实例属性和实例方法的实现来丰富实例对象的功能。

In [1]: class NewAddrBookEntry(object):               # 定义一个类
...: """New address book entry class."""
...: def __init__(self, name, phone): # 定义构造器
...: self.name = Name(name) # 创建 Name 对象
...: self.phOne= Phone(phone) # 创建 Phone 对象
...: print "Created the instance for ", self.name

上述的 class NewAddrBookEntry 有它自身和其他类 Name/Phone 组合而成, 可以实现一些更为复杂的功能。

派生

当我们希望较小的类是较大的类的组件时,组合是一个很好的处理方式。但当我们希望 相同的类却具有一些不同的功能时 派生就是最好的处理方式。这也是面向对象编程最强大的功能之一 —— 使用一个已经定义好的类,扩展它的功能或者对其进行修改生成一个新的类,但却不会对原来的类造成影响。

子类会从基类继承他们的任何属性(数据和方法),这种派生是可以继承多代的,且可以同时继承多个基类。

语法

class SubClassName(ParentClass1[, ParentClass2, ...]):
class_suite

EXAMPLE

In [8]: class Parent(object):
...: def __init__(self):
...: print "created an instance of: ", self.__class__.__name__
...:
...:

In [9]: class Child(Parent):
...: pass
...:

In [10]: c = Child()
created an instance of: Child

In [14]: p = Parent()
created an instance of: Parent

类 Child 的实例对象 c 并没有定义 __init__() 构造器,但仍然执行了 print 语句,这是因为 Child 从 Parent 继承了其构造器。

通过继承来覆盖(重载)方法

当我们派生一个子类,但同时希望相同的方法能在子类实现不同的功能,这时我们需要使用方法的 重载。使子类的方法能够覆盖父类的同名方法。

In [17]: class Parent(object):
...: def func(self):
...: print "This is Parent."
...:

In [18]: class Child(Parent):
...: def func(self):
...: print "This is Child."
...:
...:

In [19]: p = Parent()

In [20]: p.func()
This is Parent.

In [21]: c = Child()

In [22]: c.func()
This is Child.

这里子类 Child 重载了父类 Parent 的 func() 方法,实现了不同的功能。

但仍然有些场合需要我们即能使用子类的重载方法的同时,也要求我们重现父类方法的功能。那么我们可以调用那个被我们覆盖的父类方法吗?
答案是肯定的。

In [23]: Parent.func(c)
This is Parent.

我们可以通过 ParentClassName.functionName(object) 的方式来重现父类所被覆盖的方法。当然我们还有其他的方式可以实现这个效果,EG. 在子类的重载方法里显式的调用父类的同名方法:

In [24]: class Child(Parent):
...: def func(self):
...: print "This is Child."
...: Parent.func(self)

In [25]: c = Child()

In [26]: c.func()
This is Child.
This is Parent.

两种方式本质上是相同的,都是通过 父类名结合句点表达式 来实现对父类方法的调用,而且需要注意的是,在调用的时候必须传递一个实例对象给 func(),否则会触发参数不匹配的语法错误。

还有一个更好的实现方式就是子类使用 super() 内置函数:

In [27]: class Child(Parent):
...: def func(self):
...: print "This is Child."
...: super(Child, self).func()
...:

In [28]: c = Child()

In [29]: c.func()
This is Child.
This is Parent.

super() 内置函数不仅能自动的找到父类方法,并且还是自动的为父类方法传入 self 参数来实现实例方法的绑定。

最常用的重载场景(实例方法的重载)

最常用的重载场景莫过于 重载父类的构造器 了。
在上述的例子可以看出,当我们在子类中没有重载构造器的时候,会自动的调用父类的构造器。这很明显是不符合我们的需求的,因为我们常常需要在子类中定义一些新的成员属性。但是问题是:当我们为了初始化子类中新的成员属性时,不可避免的需要重复的编写初始化从父类中继承而来的属性的代码。 这也不符合代码重用的原则,所以我们一般会采用 重载构造器(init()) + 重现父类构造器(super()) 的方式来解决这个问题。

In [32]: class Parent(object):
...: def __init__(self, name, age):
...: self.name = name
...: self.age = age
...: print "This is Parent."
...:

In [33]: class Child(Parent):
...: def __init__(self, name, age, sex): # 初始化子类实例对象的属性
...: super(Child, self).__init__(name, age) # 初始化父类实例对象的属性
...: self.sex = sex
...: print "This is Child."
...:

In [35]: p = Parent("fanguiju", "24")
This is Parent.

In [36]: c = Child("fanguiju", "24", "man")
This is Parent.
This is Child.

In [37]: c.name
Out[37]: 'fanguiju'

In [38]: c.age
Out[38]: '24'

In [39]: c.sex
Out[39]: 'man'

一般而言,我们会在子类的构造器中首先调用父类的构造器,当然这并不是强制的。只是为了我们能够在执行子类构造器的代码之前首先完成对父类属性的初始化,防止在调用从父类继承而来的属性时仍未初始化的问题出现。

使用 super() 内置函数的漂亮之处在于,我们不需要明确的给出父类的名字,交由解析器去自动的找到该子类的父类,并自动的传入 self 参数来完成绑定。这样能够让代码具有更高的灵活性,我们只需要改变子类的定义语句,就可以改变类的继承关系。

从标准类中派生(类方法的重载)

不可变数据类型的派生:定义一个精度为 2 的浮点数据类型
派生不可变标准类,经常需要重载类方法,而类方法的重载一般是重载 __new__(),也就是所谓的 真·构造器

class RoundFloat(float):
def __new__(cls, value):
return float.__new__(cls, round(value, 2)) # 将类对象 float 传入参数 cls

In [44]: RoundFloat(1.1111111)
Out[44]: 1.11

真·构造器会自动的将类对象 RoundFloat 传入 cls 参数,类似于构造器__init__(self)

可变数据类型的派生:定义一个有序的字典数据类型
可变数据类型的派生可能不需要使用 构造器 或者 真·构造器 也能够实现。

class SortedDict(dict):
def keys(self):
return sorted(super(SortedDict, self).keys())

In [47]: for x in SortedDict((("name", "fanguiju"), ("age", 24), ("sex", "man"))):
...: print x
...:
age
name
sex

通过 SortedDict 生成的字典按照字母的顺序排序。

需要注意的是: 在 Python 2.2 之后将类和类型合并了,所以所有的数据类型都是一个类,反之我们定义了一个类也相当于定义了一个新的类型。


推荐阅读
  • 提升Python编程效率的十点建议
    本文介绍了提升Python编程效率的十点建议,包括不使用分号、选择合适的代码编辑器、遵循Python代码规范等。这些建议可以帮助开发者节省时间,提高编程效率。同时,还提供了相关参考链接供读者深入学习。 ... [详细]
  • 本文介绍了lua语言中闭包的特性及其在模式匹配、日期处理、编译和模块化等方面的应用。lua中的闭包是严格遵循词法定界的第一类值,函数可以作为变量自由传递,也可以作为参数传递给其他函数。这些特性使得lua语言具有极大的灵活性,为程序开发带来了便利。 ... [详细]
  • GetWindowLong函数
    今天在看一个代码里头写了GetWindowLong(hwnd,0),我当时就有点费解,靠,上网搜索函数原型说明,死活找不到第 ... [详细]
  • Python实现变声器功能(萝莉音御姐音)的方法及步骤
    本文介绍了使用Python实现变声器功能(萝莉音御姐音)的方法及步骤。首先登录百度AL开发平台,选择语音合成,创建应用并填写应用信息,获取Appid、API Key和Secret Key。然后安装pythonsdk,可以通过pip install baidu-aip或python setup.py install进行安装。最后,书写代码实现变声器功能,使用AipSpeech库进行语音合成,可以设置音量等参数。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • 使用Ubuntu中的Python获取浏览器历史记录原文: ... [详细]
  • 闭包一直是Java社区中争论不断的话题,很多语言都支持闭包这个语言特性,闭包定义了一个依赖于外部环境的自由变量的函数,这个函数能够访问外部环境的变量。本文以JavaScript的一个闭包为例,介绍了闭包的定义和特性。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • 如何使用Java获取服务器硬件信息和磁盘负载率
    本文介绍了使用Java编程语言获取服务器硬件信息和磁盘负载率的方法。首先在远程服务器上搭建一个支持服务端语言的HTTP服务,并获取服务器的磁盘信息,并将结果输出。然后在本地使用JS编写一个AJAX脚本,远程请求服务端的程序,得到结果并展示给用户。其中还介绍了如何提取硬盘序列号的方法。 ... [详细]
  • Voicewo在线语音识别转换jQuery插件的特点和示例
    本文介绍了一款名为Voicewo的在线语音识别转换jQuery插件,该插件具有快速、架构、风格、扩展和兼容等特点,适合在互联网应用中使用。同时还提供了一个快速示例供开发人员参考。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • 不同优化算法的比较分析及实验验证
    本文介绍了神经网络优化中常用的优化方法,包括学习率调整和梯度估计修正,并通过实验验证了不同优化算法的效果。实验结果表明,Adam算法在综合考虑学习率调整和梯度估计修正方面表现较好。该研究对于优化神经网络的训练过程具有指导意义。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • 本文介绍了如何使用python从列表中删除所有的零,并将结果以列表形式输出,同时提供了示例格式。 ... [详细]
author-avatar
Jason
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有