python物件導向程式設計之組合

任平生78發表於2017-07-07

前面講了面向類與物件的繼承,知道了繼承是一種什麼“是”什麼的關係。

然而類與類之間還有另一種關係,這就是組合

先來看兩個例子:
先定義兩個類,一個老師類,老師類有名字,年齡,出生的年,月和日,所教的課程等特徵以及走路,教書的技能。

class Teacher:
    def __init__(self,name,age,year,mon,day):
        self.name=name
        self.age=age
        self.year=year
        self.mon=mon
        self.day=day

    def walk(self):
        print("%s is walking slowly"%self.name)

    def teach(self):
        print("%s is teaching"%self.name)

再定義一個學生類,學生類有名字,年齡,出生的年,月和日,學習的組名等特徵以及走路,學習的技能

class Student:
    def __init__(self,name,age,year,mon,day):
        self.name=name
        self.age=age
        self.year=year
        self.mon=mon
        self.day=day

    def walk(self):
        print("%s is walking slowly"%self.name)

    def study(self):
        print("%s is studying"%self.name)

根據類的繼承這個特性,可以把程式碼縮減一下。

定義一個人類,然後再讓老師類和學生類繼承人類的特徵和技能:

class People:
    def __init__(self,name,age,year,mon,day):
        self.name=name
        self.age=age
        self.year=year
        self.mon=mon
        self.day=day

    def walk(self):
        print("%s is walking"%self.name)

class Teacher(People):
    def __init__(self,name,age,year,mon,day,course):
        People.__init__(self,name,age,year,mon,day)
        self.course=course

    def teach(self):
        print("%s is teaching"%self.name)

class Student(People):
    def __init__(self,name,age,year,mon,day,group):
        People.__init__(self,name,age,year,mon,day)
        self.group=group

    def study(self):
        print("%s is studying"%self.name)

再對老師和學生進行例項化,得到一個老師和一個學生。

t1=Teacher("alex",28,1989,9,2,"python")
s1=Student("jack",22,1995,2,8,"group2")

現在想知道t1和s1的名字,年齡,出生的年,月,日都很容易,但是想一次性列印出
t1或s1的生日就不那麼容易了,這時就需要用字串進行拼接了,有沒有什麼更好的辦法呢??

那就是組合。

繼承是一個子類是一個父類的關係,而組合則是一個類有另一個類的關係。

可以說每個人都有生日,而不能說人是生日,這樣就要使用組合的功能 。
可以把出生的年月和日另外再定義一個日期的類,然後用老師或者是學生與這個日期的類
組合起來,就可以很容易得出老師t1或者學生s1的生日了,再也不用字串拼接那麼麻煩了。
來看下面的程式碼:

class Date:
    def __init__(self,year,mon,day):
        self.year=year
        self.mon=mon
        self.day=day

    def birth_info(self):
        print("The birth is %s-%s-%s"%(self.year,self.mon,self.day))

class People:
    def __init__(self,name,age,year,mon,day):
        self.name=name
        self.age=age
        self.birth=Date(year,mon,day)

    def walk(self):
        print("%s is walking"%self.name)

class Teacher(People):
    def __init__(self,name,age,year,mon,day,course):
        People.__init__(self,name,age,year,mon,day)
        self.course=course

    def teach(self):
        print("%s is teaching"%self.name)

class Student(People):
    def __init__(self,name,age,year,mon,day,group):
        People.__init__(self,name,age,year,mon,day)
        self.group=group

    def study(self):
        print("%s is studying"%self.name)
t1=Teacher("alex",28,1989,9,2,"python")
s1=Student("jack",22,1995,2,8,"group2")

這樣一來,可以使用跟前面一樣的方法來呼叫老師t1或學生s1的姓名,年齡等特徵
以及走路,教書或者學習的技能。

print(t1.name)
t1.walk()
t1.teach()

輸出為:

alex    
alex is walking
alex is teaching

那要怎麼能夠知道他們的生日呢:

print(t1.birth)

輸出為:

<__main__.Date object at 0x0000000002969550>

這個birth子類Teacher從父類People繼承過來的,而父類People的birth又是與Date這個類組合在一
起的,所以,這個birth是一個物件。而在Date類下面有一個birth_info的技能,這樣就可以通過
呼叫Date下面的birth_info這個函式屬性來知道老師t1的生日了。

t1.birth.birth_info()

得到的結果為:

The birth is 1989-9-2

同樣的,想知道例項學生s1的生日也用同樣的方法:

s1.birth.birth_info()

得到的結果為:

The birth is 1995-2-8

組合就是一個類中使用到另一個類,從而把幾個類拼到一起。組合的功能也是為了減少重複程式碼。


相關文章