準備工作
為了確保類別是新類型,應該把 _metaclass_=type 入到你的模組的最開始。
class NewType(Object): mor_code_here class OldType: mor_code_here
在這個兩個類別中NewType是新類,OldType是屬於舊類,如果前面加上 _metaclass_=type ,那麼兩個類別都屬於新類別。
構造方法
構造方法與其的方法不一樣,當一個物件被創建會立即呼叫構造方法。創建一個python的建構方法很簡答,只要把init方法,從簡單的init方法,轉換成魔法版本的_init_方法就可以了。
class FooBar: def __init__(self): self.somevar = 42 >>> f =FooBar() >>> f.somevar 42
重寫一個一般方法
每一個類別都可能擁有一個或多個超類別(父類別),它們從超類別中繼承行為方法。
class A: def hello(self): print 'hello . I am A.' class B(A): pass >>> a = A() >>> b = B() >>> a.hello() hello . I am A.
因為B類沒有hello方法,B類繼承了A類,所以會呼叫A 類的hello方法。
在子類別中增加功能功能的最基本的方式就是增加方法。但是也可以重寫一些超類別的方法來自訂繼承的行為。如下:
class A: def hello(self): print 'hello . I am A.' class B(A): def hello(self): print 'hello . I am B' >>> b = B() >>> b.hello() hello . I am B
特殊的和構造方法
重寫是繼承機制中一個重要的內容,對一於構造方法尤其重要。看下面的例子:
class Bird: def __init__(self): self.hungry = True def eat(self): if self.hungry: print 'Aaaah...' self.hungry = False else: print 'No, thanks!' >>> b = Bird() >>> b.eat() Aaaah... >>> b.eat() No, thanks!
這個類中定義了鳥有吃的能力, 當牠吃過一次後再次就會不餓了,透過上面的執行結果可以清晰的看到。
那麼用SongBird類來繼承Bird 類,並且給它添加歌唱的方法:
class Bird: def __init__(self): self.hungry = True def eat(self): if self.hungry: print 'Aaaah...' self.hungry = False else: print 'No, thanks!' class SongBird(Bird): def __init__(self): self.sound = 'Squawk!' def sing(self): print self.sound >>> s = SongBird() >>> s.sing() Squawk! >>> s.eat() Traceback (most recent call last): File "<pyshell#26>", line 1, in <module> s.eat() File "C:/Python27/bird", line 6, in eat if self.hungry: AttributeError: 'SongBird' object has no attribute 'hungry'
異常很清楚地說明了錯誤:SongBirdhungry原因是這樣的:在SongBird中,構造方法被重寫,但新的構造方法沒有任何關於初始化hungry特性的程式碼。為了達到預期的效果,SongBird的建構方法必須呼叫其超類別Bird的建構方法來確保進行基本的初始化。
兩種方法實作:
一 、呼叫未綁定的超類別建構方法
class Bird: def __init__(self): self.hungry = True def eat(self): if self.hungry: print 'Aaaah...' self.hungry = False else: print 'No, thanks!' class SongBird(Bird): def __init__(self): Bird.__init__(self) self.sound = 'Squawk!' def sing(self): print self.sound >>> s = SongBird() >>> s.sing() Squawk! >>> s.eat() Aaaah... >>> s.eat() No, thanks!
碼
__metaclass__ = type #表明为新式类 class Bird: def __init__(self): self.hungry = True def eat(self): if self.hungry: print 'Aaaah...' self.hungry = False else: print 'No, thanks!' class SongBird(Bird): def __init__(self): super(SongBird,self).__init__() self.sound = 'Squawk!' def sing(self): print self.sound >>> s.sing() Squawk! >>> s.eat() Aaaah... >>> s.eat() No, thanks!
class Rectangle: def __init__(self): self.width = 0 self.height = 0 def setSize(self,size): self.width , self.height = size def getSize(self): return self.width , self.height >>> r = Rectangle() >>> r.width = 10 >>> r.height = 5 >>> r.getSize() (10, 5) >>> r.setSize((150,100)) >>> r.width 150
__metaclass__ = type class Rectangle: def __int__(self): self.width = 0 self.height = 0 def setSize(self,size): self.width, self.height = size def getSize(self): return self.width ,self.height size = property(getSize ,setSize) >>> r = Rectangle() >>> r.width = 10 >>> r.height = 5 >>> r.size (10, 5) >>> r.size = 150,100 >>> r.width 150中,property 函數建立了一個屬性,其中存取器函數被用作參數(先取值,然後是賦值),這個屬性命為size 。這樣一來就不再需要擔心是怎麼實現的了,可以用同樣的方式處理width、height 和size。