上一課舉了一個(gè)面向?qū)ο蠛兔嫦蜻^程相比較的例子之后,有些同學(xué)表示,仍然沒太看出面向?qū)ο蟮膬?yōu)勢。沒關(guān)系,那是因?yàn)槲覀儸F(xiàn)在接觸的程序還不夠復(fù)雜,等以后你寫的程序越來越大,就能體會(huì)到這其中的差別了。
今天我們就來舉一個(gè)稍稍再復(fù)雜一點(diǎn)的例子。
仍然是從A地到B地,這次除了有汽車,我們還有了一輛自行車!
自行車和汽車有著相同的屬性:速度(speed)。還有一個(gè)相同的方法(drive),來輸出行駛/騎行一段距離所花的時(shí)間。但這次我們要給汽車增加一個(gè)屬性:每公里油耗(fuel)。而在汽車行駛一段距離的方法中,除了要輸出所花的時(shí)間外,還要輸出所需的油量。
面向過程的方法,你可能需要寫兩個(gè)函數(shù),然后把數(shù)據(jù)作為參數(shù)傳遞進(jìn)去,在調(diào)用的時(shí)候要搞清應(yīng)該使用哪個(gè)函數(shù)和哪些數(shù)據(jù)。有了面向?qū)ο?,你可以把相關(guān)的數(shù)據(jù)和方法封裝在一起,并且可以把不同類中的相同功能整合起來。這就需要用到面向?qū)ο笾械牧硪粋€(gè)重要概念:繼承。
我們要使用的方法是,創(chuàng)建一個(gè)叫做Vehicle的類,表示某種車,它包含了汽車和自行車所共有的東西:速度,行駛的方法。然后讓Car類和Bike類都繼承這個(gè)Vehicle類,即作為它的子類。在每個(gè)子類中,可以分別添加各自獨(dú)有的屬性。
Vehicle類被稱為基本類或超類,Car類和Bike類被成為導(dǎo)出類或子類。
1. class Vehicle:
2. def __init__(self, speed):
3. self.speed = speed
4.
5. def drive(self, distance):
6. print 'need %f hour(s)' %(distance / self.speed)
7.
8. class Bike(Vehicle):
9. pass
10.
11. class Car(Vehicle):
12. def __init__(self, speed, fuel):
13. Vehicle.__init__(self, speed)
14. self.fuel = fuel
15.
16. def drive(self, distance):
17. Vehicle.drive(self, distance)
18. print 'need %f fuels' % (distance *self.fuel)
19.
20. b = Bike(15.0)
21. c = Car(80.0, 0.012)
22. b.drive(100.0)
23. c.drive(100.0)
24.
解釋一下代碼:
__init__函數(shù)會(huì)在類被創(chuàng)建的時(shí)候自動(dòng)調(diào)用,用來初始化類。它的參數(shù),要在創(chuàng)建類的時(shí)候提供。于是我們通過提供一個(gè)數(shù)值來初始化speed的值。
class定義后面的括號里表示這個(gè)類繼承于哪個(gè)類。Bike(Vehicle)就是說Bike是繼承自Vehicle中的子類。Vehicle中的屬性和方法,Bike都會(huì)有。因?yàn)锽ike不需要有額外的功能,所以用pass在類中保留空塊,什么都不用寫。
Car類中,我們又重新定義了__init__和drive函數(shù),這樣會(huì)覆蓋掉它繼承自Vehicle的同名函數(shù)。但我們依然可以通過“Vehicle.函數(shù)名”來調(diào)用它的超類方法。以此來獲得它作為Vehicle所具有的功能。注意,因?yàn)槭峭ㄟ^類名調(diào)用方法,而不是像之前一樣通過對象來調(diào)用,所以這里必須提供self的參數(shù)值。在調(diào)用超類的方法之后,我們又給Car增加了一個(gè)fuel屬性,并且在drive中多輸出一行信息。
最后,我們分別創(chuàng)建一個(gè)速度為15的自行車對象,和一個(gè)速度為80、耗油量為0.012的汽車,然后讓它們?nèi)バ旭?00的距離。