Python 面向对象编程容易忽视的知识点

gu311711 8年前
   <p><img alt="Python 面向对象编程容易忽视的知识点" src="https://simg.open-open.com/show/ea05b3c2f691cd8ef28db8e25328f436.png"></p>    <p>类既可以很简单,也可以很复杂。最简单的情况,类仅用作名称空间,这意味着你把数据保存在变量中,对他们按照名称空间进行分组,使他们出入同样的关系空间中。这样的类仅作为容器对象来共享名字空间。</p>    <p>1. __init__(self,...)‘构造器’方法</p>    <p>__init__(self,...)方法实际上并不是一个构造器,其并没有创造一个新的对象,Python通过函数操作符()创建对象。在解释器创建一个实例后,最先调用__init__(self,...)方法,定义额外的行为,如果没有定义或者覆盖__init__(self,...)方法,对实例不会施加任何特别的操作,直接返回它的对象,实例化过程完毕。函数的返回值应当为None,如果返回来任意对象,会导致TypeError异常。</p>    <pre>  <code class="language-python">class ReturnInt:      def __init__(self):          return 1    ri=ReturnInt()    Traceback (most recent call last):    File "C:/Users/PycharmProjects/untitled/Study EXP/Q.py", line 5, in <module>      ri=ReturnInt()  TypeError: __init__() should return None</code></pre>    <p>重写子类的__init__(self,...)方法不会自动调用基类的__init__(self,...)。所以如果需要在子类中调用基类的__init__(self,...),需要明确指出。</p>    <pre>  <code class="language-python">class C(P):    def __init__(self):      P.__init__(self)</code></pre>    <p>我们可以使用super()函数更方便高效的重写上面的代码:</p>    <pre>  <code class="language-python">class C(P):    def __init__(self):      super(C,self).__init__()</code></pre>    <p>不需要提供明确的父类,super()函数会帮助我们找到相应的父类,然后方便调用相关的属性。</p>    <p>2. __new__(cls,...)‘构造器’方法</p>    <p>调用__new__(cls,...)时,实例不一定已经被创建,与__init__(self,...)方法需要传入实例的引用不同,__new__(cls,...)传入当前类的引用。可以用来对内建类型进行派生,实例化不可变对象。</p>    <p>3. 类属性</p>    <p>类属性仅与其被定义的的类相绑定,主要包括数据属性(静态变量)和方法。方法在类中定义,但却只能被实例调用。如果没有与类的数据属性同名的实例属性,通过实例也可以访问类的数据属性,但是却不能作修改。</p>    <p>特殊的类属性:</p>    <pre>  <code class="language-python">C.__name__    类C的名字(字符串)  C.__doc__     类C的文档字符串  C.__bases__   类C的所有父类构成的元组  C.__dict__    类C的属性  C.__module__  类C定义所在的模块  C.__class__   实例C对应的类</code></pre>    <p>4. 实例属性</p>    <p>实例仅拥有数据属性,即类属性。实例属性可以动态创建,但是如果属性在条件语句中创建,而该条件语句未执行,则该属性实际并不存在,如果在后面代码访问该属性,就会出错。</p>    <p>带默认参数的__init__(self,...)可以更有效的初始化一个实例,这样可以省去显式传值的麻烦,但是默认参数应当是不变的对象,在使用入列表和字典等可变对象时应时刻保持警惕。</p>    <pre>  <code class="language-python">class demo_list:    def __init__(self, l=[]):       self.l = l     def add(self, ele):       self.l.append(ele)  def appender(ele):     obj = demo_list()     obj.add(ele)     print obj.lif   __name__ == "__main__":     for i in range(5):       appender(i)      [0]   [0, 1]  [0, 1, 2]  [0, 1, 2, 3]   [0, 1, 2, 3, 4]</code></pre>    <p>因为<strong>默认参数只会计算一次,不会重复使用</strong>,在上面的例子中,虽然使用不同的新建实例,可因为构造器的参数使用的是对列表的引用作为默认参数,所以每次的实例属性都指向该列表所在的空间。<br> <a href="http://www.open-open.com/lib/view/open1466673893029.html">点击查看更多相关资料</a></p>    <p>使用内建函数<strong>dir()</strong>可以显示类和实例属性,实例具有__dict__的特殊属性,该属性由字典组成,包含一个实例的所有属性。</p>    <pre>  <code class="language-python">'#的意思为,当ReturnInt为新式类时,dir对应的结果'    class ReturnInt:    #class ReturnInt(object):    def __init__(self):            pass  ri=ReturnInt()  ri.name='a'  ri.age=10  print ri.__dict__  print dir(ri)    {'age': 10, 'name': 'a'}  ['__doc__', '__init__', '__module__', 'age', 'name']    #['__class__', '__delattr__', '__dict__', '__doc__', '__format__', '__getattribute__', '__hash__', '__init__', '__module__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'age', 'name']</code></pre>    <p>从上面的程序中可以看到除了我们添加的实例属性,实例还有许多其他的属性,这些属性是实例的内建类型属性。</p>    <p>5. 绑定和方法调用</p>    <p>类属性之一的方法只有在其所属的类拥有实例时,才能被调用,当存在一个实例时,方法才被认为是绑定到那个实例了,没有实例时,方法就是未绑定的。</p>    <p>当类具有实例时,比如类MyClass,通过mc.foo()调用可以把mc即self自动传入到foo()中,而不需要我们明确的写为mc.foo(self)。当时当没有一个实例并且需要调用一个非绑定的方法时就必须传递self参数。</p>    <p><br> 文/<a href="/misc/goto?guid=4959674774366778768">CinderellaM</a>(简书)</p>