Pythoner_MLer

  博客园 :: 首页 :: 博问 :: 闪存 :: 新随笔 :: 联系 :: 订阅 订阅 :: 管理 ::

希望通过博客园持续的更新,分享和记录Python基础知识到高级应用的点点滴滴!

第二波:第7章  更加抽象

[总览] 创建自己的对象,是Python的核心概念!Python被称为面向对象的语言。介绍如何创建对象,以及多态、封装、方法、特性、超类以及继承的概念。

 

[7.1] 对象的魔力

   面向对象程序设计中的术语对象(object)基本上可以看做数据(特性)以及由一系列可以存取、操作这些数据的方法所组成的集合。对象最重要的优点包括以下几方面:

  多态polymorphism:意味着可以对不同类的对象使用同样的操作。

  封装encapsulation:对外部世界隐藏对象的工作细节。

  继承inheritance:以普通的类为基础简历专门的类对象。

  面向对象程序设计最有趣的特性是多态。

[7.1.1] 多态

  多态意味着就算不知道变量所引用的对象类型是什么,还是能对它进行操作,而它也会根据对象(或类)类型的不同而表现出不同的行为。

  1. 多态和方法

  绑定到对象特性上面的函数称为方法method。

  2. 多态的多种形式

  任何不知道对象到底是什么类型,但是又要对对象“做点什么”的时候,都会用到多态。很多内建运算符和函数都有多态的性质。唯一能够毁掉多态的就是使用函数显示地检查类型,比如type、isinstance以及issubclass函数等。

[7.1.2] 封装

  封装是对全局作用域中其他区域隐藏多余信息的原则。和多态一样,它们都会帮助处理程序组件而不用过多关心多余细节。

  封装不等同于多态。多态可有让用户对于不知道什么类(或对象类型)的对象进行方法调用,而封装是可以不用关心对象是如何构造的而直接进行使用。

[7.1.3] 继承

  比如已经有了一个类,而又想建立一个非常类似的类,新的类肯那个只是添加几个方法。在编写新类时,又不想把旧类的代码全部复制故去,此时将使用到类的继承。

[7.2] 类和类型

  什么是类,以及它和类型又有什么不同或相同。

[7.2.1] 类到底是什么

   类是一种对象。所有的对象都属于某一个类,称为类的实例instance。例如,鸟就是“鸟类”的实例。这就是一个有很多子类的一般类。可以将“鸟类”想象成所有鸟的集合,而”百灵鸟类“是其中的一个子集。当一个对象所属的类时另外一个对象所属类的子集时,前者就被称为后者的子类subclass,所以”百灵鸟类“是”鸟类“的子类。相反,”鸟类“是”百灵鸟类“的超类superclass。

  Python中,习惯上都使用单数名词,并且首字母大写。

  定义子类只是定义更多的方法的过程。

[7.2.2] 创建自己的类

  __metaclass__=type

  class Person:

    def setName(self,name):

      self.name=name

    def getName(self):

      return self.name

    def greet(self):

      print "Hello,World!I'm %s." % self.name

  新式类的语法中,需要在模块或者脚本开始的地方放置赋值语句__metaclass__=type。

  这个例子包含3个方法定义,除了它们是写在class语句里面外,一切都像是函数定义。Person是类的名字。class语句会在函数定义的地方创建子集的命名空间。self参数看起来有点奇怪,self它是对于对象自身的引用。在调用foo的setName和greet函数时,foo自动将自己作为第一个参数传入函数中,因此形象地命名为self。因为它总是对象自身,所以习惯上总是叫self。

  特性是可以在外部访问的。

  foo.name

  如果知道foo是Person的实例的话,那么嗨可以把foo.greet()看做Person.greet(foo)方便的简写。

[7.2.3】 特性、函数和方法

  self参数事实上正是方法和函数的区别。方法将它们的第一个参数绑定到所属的实例上。因此这个参数可以不必提供。

  【ToDo:再论私有化】

  Python并不直接支持私有方式,而要靠程序员自己把握izai外部进行特性修改的时机。毕竟在使用对象前应该知道如何使用,但是可以用一些小技巧达到私有特性的效果。

  为了让方法或特性变为私有(从外部无法访问),只要在它的名字前面加上双下划线即可:

  class Secretive:

    def __inaccessible(self):

      print "Bet you can't see me..."

    def accessible(self):

      print "The secret message is:"

      self.__inacessible()

  现在__inacessible从外界是无法访问的,而在类内部还能使用访问。类的内部定义中,所有以双下划线开始的名字都被”翻译“成前面加上单下划线和雷鸣的形式。

  如果不需要使用这种方法,但是又想让其他对象不要访问内部数据,那么可以使用但下划线。前面有下划线的名字都不会被带有型号的import语句(from module import *)导入。

[7.2.4] 类的命名空间

  所有位于class语句中的代码都在特殊的命名空间中执行---类命名空间class namespace。这个命名空间可由类内部所有成员访问。类的定义其实就是执行代码块。

[7.2.5] 指定超累

  子类可以扩展超类的定义。将其他雷鸣写在class语句后的圆括号内可以指定超类。

  class Filter:

    def init(self):

      self.blocked=[]

    def filter(self,sequence):

      return [x for x in sequence if x not in self.blocked]

  class SPAMFilter(Filter):  #SPAMFilter是Filter的子类

    def init(self): #重写Filter超类中的init方法

      self.blocked=['SPAM']

  注意SPAMFilter定义的两个要点:

    这里用提供新定义的方式重写了Filter的init定义。

    filter方法的定义是从Filter类中继承的,所以不用重写它的定义。 

  第二个要点揭示了继承的用处。

[7.2.6] 调查继承

  如果想要查看一个类是否是另一个类的子类,可以使用内建的issubclass函数:

  issubclass(SPAMFilter,Filter)

  如果想要知道已知类的基类,可以直接使用它的特殊特性__bases__:

  SPAMFilter.__bases__

  同样,还能使用isinstance方法检查一个对象是否是一个类的实例:

  isinstance(s,SPAMFilter)

[7.2.7] 多个超累

  __bases__这个复数形式,暗示它的基类可能会多于一个。

  子类从自己的超类继承所有的行为。称为多重继承(multiple inheritance),是个非常有用的工具。

  当使用多重继承时,有个需要注意的地方。如果一个方法从多个超类继承,那么必须要注意一下超类的顺序:先继承的类中的方法会重写后继承的类中的方法。如果超类们共享一个超类,那么在查找给定方法或者特性时访问超类的顺序称为MRO(Method Resolution Order,方法判定顺序)。

[7.2.8] 接口和内省

  ”接口“的概念与多态有关。在处理多态对象时,只要关心它的接口(或称”协议“)即可---也就是公开的方法和特性。在Python中,不用显示地制定对象必须包含哪些方法才能作为参数接收。

  可以检查所需方法是否已经存在:

  hasattr(tc,'talk')

  上面代码中,对象tc有个叫做talk的特性,但是没有fnord特性。如果需要的话,甚至还能检查talk特性是否可调用:

  callable(getattr(tc,'talk',None))

[7.3] 一些关于面向对象设计的思考

  给出一些要点:

    将属于一类的对象放在一起。如果一个函数操纵一个全局变量,那么两者最好都在类内部作为特性和方法出     现。
    不要让对象过于亲密。方法应该只关心自己实例的特性。

    要小心继承,尤其是多重继承。

    简单就好,多数方法都应能在30秒内被读完,将代码行控制在一页或一屏内。

  当考虑需要什么类以及类要有什么方法时,应该尝试下面的方法:

  (1)写下问题的描述(程序要做什么?),把所有名词、动词和形容词加下划线。

  (2)对于所有名词,用做可能的类。

  (3)对于所有动词,用作可能的方法。

  (4)对于所有形容词,用作可能的特性。

  (5)把所有方法和特性分配到类。

  这就是面向对象类型的草图。还可以考虑类和对象之间的关系以及它们的作用,可以用以下步骤精炼模型:

  (1)写下一系列的使用实例,也就是程序应用时的场景,试着包括所有的功能。

  (2)考虑每个使用实例,保证模型包括所有需要的东西。

【7.4】 小结

  对象:对象包括特性和方法。特性只是作为对象的一部分的变量,方法则是存储在对象内的函数。方法和其他函数的区别在于方法总是将对象作为自己的第一个参数,这个参数一般称为self。

  类:类代表对象的集合,每个对象都有一个类。类的主要任务是定义它的实例会用到的方法。

  多态:多态是实现将不同类型和类的对象进行同样对待的特性。不需要知道对象属于哪个类就能调用方法。

  封装:对象可以将它们的内部状态隐藏起来。

  继承:一个类可以使一个或者多个类的子类。子类从超类继承所有方法。

  接口和内省:需要知道对象到底有什么方法和特性,有些函数可以帮助完成这项工作。

  面向对象设计:创建容易理解的设计是很重要的。

 

posted on 2016-07-04 00:50  Pythoner_MLer  阅读(252)  评论(0编辑  收藏  举报