跟老齐学Python之编写类之三子类


Posted in Python onOctober 11, 2014

关于类,看官想必已经有了感觉,看下面的代码,请仔细阅读,并看看是否能够发现点什么问题呢?

#!/usr/bin/env python

#coding:utf-8
class Person:

    def __init__(self, name, lang, email):

        self.name = name

        self.lang = lang

        self.email = email
    def author(self):

        return self.name
class Programmer:

    def __init__(self, name, lang, email, system, website):

        self.name = name

        self.lang = lang

        self.email = email

        self.system = system

        self.website = website
    def pythoner(self):

        pythoner_list = [ self.name, self.lang, self.email, self.system, self.website ]

        return pythoner_list
if __name__=="__main__":

    writer = Person("qiwsir","Chinese","qiwsir@gmail.com")

    python = Programmer("qiwsir","Python","qiwsir@gmail.com","Ubutun","qiwsir.github.io")

    print "My name is:%s"%writer.author()

    print "I write program by:%s"%python.pythoner()[1]

 上面这段代码,运行起来没有什么问题,但是,仔细看,发现有两个类,一个名字叫做Person,另外一个叫做Programmer,这还不是问题所在,问题所在是这两个类的构造函数中,存在这相同的地方:self.name=name,self.lang=lang,self.email=email,这对于追求代码质量的程序员,一般是不允许的。最好不要有重复代码或者冗余代码。可是,在两个类中都要有这些参数,应该怎么办呢?

子类、父类和继承

看下面的代码,里面有两个类A,B。这段程序能够正确运行,每个类的功能是仅仅打印指定的内容。

#!/usr/bin/env python

#coding:utf-8
class A:

    def __init__(self):

        print "aaa"
class B:

    def __init__(self):

        print "bbb"
if __name__=="__main__":

    a = A()

    b = B()
#运行结果

aaa

bbb

 上面的两个类彼此之间没有所谓的父子关系。现在稍加改变,将类B改写,注意观察与上面的差异。

#!/usr/bin/env python

#coding:utf-8
class A:

    def __init__(self):

        print "aaa"
class B(A):         #这里和上面程序不同。B继承了A

    def __init__(self):

        print "bbb"
if __name__=="__main__":

    a = A()

    b = B()
#运行结果

aaa

bbb

 这段程序中,类B跟前面的那段有一点不同,class B(A):,这样写就表明了B相对A的关系:B是A的子类,B从A继承A的所有东西(子承父业)。

但是,看官发现了没有,运行结果一样。是的,那是以为在B中尽管继承了A,但是没有调用任何A的东西,就好比儿子从老爸那里继承了财富,但是儿子一个子也没动,外界看到的和没有继承一样。

#!/usr/bin/env python

#coding:utf-8
class A:

    def __init__(self):

        print "aaa"
class B(A):

    def __init__(self):

        #print "bbb"

        A.__init__(self)    #运行继承的父类
if __name__=="__main__":

    a = A()

    b = B()
#运行结果

aaa

aaa

 这回运行结果有了变化,本来b=B()是运行类B,但是B继承了A,并且在初始化的构造函数中,引入A的构造函数,所以,就运行A的结果相应结果了。

下面把最开头的那端程序用子类继承的方式重写,可以是这样的:

#!/usr/bin/env python

#coding:utf-8
class Person:

    def __init__(self, name, lang, email):

        self.name = name

        self.lang = lang

        self.email = email
    def author(self):

        return self.name

"""

class Programmer:

    def __init__(self, name, lang, email, system, website):

        self.name = name

        self.lang = lang

        self.email = email

        self.system = system

        self.website = website
    def pythoner(self):

        pythoner_list = [ self.name, self.lang, self.email, self.system, self.website ]

        return pythoner_list

"""
class Programmer(Person):       #继承父类Person

    def __init__(self, name, lang, email, system, website):

        Person.__init__(self,name,lang,email)   #将Person.__init__()的功能继承到这里

        #self.name = name                       #这三句是Person中已经搞定的,就不用重复

        #self.lang = lang                       #通过继承已经实现了这三句的功能

        #self.email = email

        self.system = system                    #子类中不同于Person父类部分

        self.website = website
    def pythoner(self):

        pythoner_list = [ self.name, self.lang, self.email, self.system, self.website ]

        return pythoner_list
if __name__=="__main__":

    writer = Person("qiwsir","Chinese","qiwsir@gmail.com")

    python = Programmer("qiwsir","Python","qiwsir@gmail.com","Ubutun","qiwsir.github.io")

    print "My name is:%s"%writer.author()

    print "I write program by:%s"%python.pythoner()[1]

 代码运行结果与前面一样。

列位是否理解了子类和父类、继承的特点。如果你有一个老爹,是一个高官或者富豪,那么你就官二代或者富二代了,你就从他们那里继承了很多财富,所以生活就不用太劳累了。这就是继承的作用。在代码中,也类似,继承能够让写代码的少劳累一些。

对于为什么要用继承,好友@令狐虫 大侠给了以非常精彩的解释:

从技术上说,OOP里,继承最主要的用途是实现多 态。对于多态而言,重要的是接口继承性,属性和行为是否存在继承性,这是不一定的。事实上,大量工程实践表明,重度的行为继承会导致系统过度复杂和臃肿, 反而会降低灵活性。因此现在比较提倡的是基于接口的轻度继承理念。这种模型里因为父类(接口类)完全没有代码,因此根本谈不上什么代码复用了。
在Python里,因为存在Duck Type,接口定义的重要性大大的降低,继承的作用也进一步的被削弱了。

另外,从逻辑上说,继承的目的也不是为了复用代码,而是为了理顺关系。
 
我表示完全赞同上述解释。不过看官如果不理解,也没有关系,上述解释中的精神,的确需要在编程实践中感悟才能领会到的。
Python 相关文章推荐
用Python代码来绘制彭罗斯点阵的教程
Apr 03 Python
Python中的XML库4Suite Server的介绍
Apr 14 Python
使用Python编写vim插件的简单示例
Apr 17 Python
讲解Python中fileno()方法的使用
May 24 Python
Python找出9个连续的空闲端口
Feb 01 Python
Window环境下Scrapy开发环境搭建
Nov 18 Python
python实现五子棋小程序
Jun 18 Python
python hough变换检测直线的实现方法
Jul 12 Python
浅谈对pytroch中torch.autograd.backward的思考
Dec 27 Python
python正则表达式匹配IP代码实例
Dec 28 Python
Python函数基本使用原理详解
Mar 19 Python
django orm模块中的 is_delete用法
May 20 Python
跟老齐学Python之编写类之二方法
Oct 11 #Python
Python中MYSQLdb出现乱码的解决方法
Oct 11 #Python
python操作mysql中文显示乱码的解决方法
Oct 11 #Python
wxPython窗口中文乱码解决方法
Oct 11 #Python
跟老齐学Python之编写类之一创建实例
Oct 11 #Python
跟老齐学Python之关于类的初步认识
Oct 11 #Python
跟老齐学Python之传说中的函数编写条规
Oct 11 #Python
You might like
php下删除一篇文章生成的多个静态页面
2010/08/08 PHP
ThinkPHP无限级分类原理实现留言与回复功能实例
2014/10/31 PHP
Laravel登录失败次数限制的实现方法
2020/08/26 PHP
jquery实现marquee效果(文字或者图片的水平垂直滚动)
2013/01/07 Javascript
Yii-自定义删除确认弹框(zyd)jquery实现代码
2013/03/04 Javascript
jQuery多项选项卡的实现思路附样式及代码
2014/06/03 Javascript
javascript中为某个元素指定事件的三种方式
2014/08/07 Javascript
jQuery实现选中弹出窗口选择框内容后赋值给文本框的方法
2015/11/23 Javascript
js停止冒泡和阻止浏览器默认行为的简单方法
2016/05/15 Javascript
Bootstrap select下拉联动(jQuery cxselect)
2017/01/04 Javascript
Ionic2开发环境搭建教程
2020/08/20 Javascript
Angular项目中$scope.$apply()方法的使用详解
2017/07/26 Javascript
angular中的cookie读写方法
2017/08/02 Javascript
详解如何构建一个Angular6的第三方npm包
2018/09/07 Javascript
微信小程序封装自定义弹窗的实现代码
2019/05/08 Javascript
17道题让你彻底理解JS中的类型转换
2019/08/08 Javascript
用实例说明python的*args和**kwargs用法
2013/11/01 Python
Python Matplotlib库入门指南
2015/05/18 Python
Android模拟器无法启动,报错:Cannot set up guest memory ‘android_arm’ Invalid argument的解决方法
2016/07/01 Python
Sanic框架异常处理与中间件操作实例分析
2018/07/16 Python
浅谈pyqt5中信号与槽的认识
2019/02/17 Python
Python爬虫beautifulsoup4常用的解析方法总结
2019/02/25 Python
Python实现EXCEL表格的排序功能示例
2019/06/25 Python
命令行运行Python脚本时传入参数的三种方式详解
2019/10/11 Python
在python3中实现更新界面
2020/02/21 Python
jupyter notebook 参数传递给shell命令行实例
2020/04/10 Python
使用keras实现BiLSTM+CNN+CRF文字标记NER
2020/06/29 Python
CSS3动画效果回调处理详解
2014/12/10 HTML / CSS
HTML5 Canvas基本线条绘制的实例教程
2016/03/17 HTML / CSS
STUBHUB日本:购买和出售全球活动门票
2018/07/01 全球购物
UDP协议功能
2013/01/06 面试题
主管会计岗位责任制
2014/02/10 职场文书
观看《周恩来的四个昼夜》思想汇报
2014/09/12 职场文书
QT连接MYSQL数据库的详细步骤
2021/07/07 MySQL
Java 语言中Object 类和System 类详解
2021/07/07 Java/Android
element tree树形组件回显数据问题解决
2022/08/14 Javascript