一、組合
組合:組合指的是,在一個類中以另外一個類的對象(也就是實例)作為數據屬性,稱為類的組合
也就是說:一個類的屬性是另一個類的對象,就是組合
例子:
圓環是由兩個圓組成的,圓環的面積就是外圓的面積減去內圓的面積。圓環的周長就是內圓的周長加上外圓的周長,這個時候,我們首先設計一個圓形類,計算一個圓的面積和圓的周長。然后在‘圓環類’組合圓形的實例作為自己的屬性來用(這樣的目的就是為了不用在寫面積和周長的方法了,直接組合圓類的面積和方法去求解。減少了代碼的重用)

1 from math import pi 2 class Circle: 3 def __init__(self,r): 4 self.r=r 5 def perimater(self): 6 return 2*pi*self.r 7 def area(self): 8 return pi*self.r*self.r 9 # print(Circle.perimater('r',2)) 10 # print(Circle.area('r',3)) 11 12 13 class Circle_ring: #定義一個圓環類 14 def __init__(self,outside_r,inside_r): 15 outside_bijiao = max(outside_r,inside_r) 16 intside_bijiao = min(outside_r, inside_r) 17 self.outsize_circle = Circle(outside_bijiao) #實例化一個大圓形 作為self.outside_circle屬性的值 18 self.intsize_circle = Circle(intside_bijiao) #實例化一個小圓環 19 def area(self): 20 return self.outsize_circle.area()-self.intsize_circle.area() 21 def perimater(self): 22 return self.intsize_circle.perimater()+self.outsize_circle.perimater() 23 24 25 r1 = Circle_ring(10,20) #實例化 26 print(r1.area()) 27 print(r1.perimater())
組合的兩種方式:1.在__init__方法里面組合
2.在外面組合

1 class BirthDate: 2 def __init__(self,year,month,day): 3 self.year=year 4 self.month = month 5 self.day = day 6 class Course: 7 def __init__(self,name,price,period): #period為周期 8 self.name =name 9 self.price = price 10 self.period = period 11 class Teacher: 12 def __init__(self,name,salary,year,month,day,price,period): #那么這個里面也要把該有的屬性傳進去 13 self.birth = BirthDate(year,month,day) #在里面組合(將BirthDate里面有的屬性傳入進去) 14 self.course=Course(name,price,period) 15 self.name = name 16 self.salary = salary 17 # 實例化方法一: 18 egg = Teacher('egon',2000,1999,12,2,'6 months','15800') #也要實例化,Teacher類里面的屬性都得實例化 19 print(egg.birth.month) #當然老師也有生日,就讓egg.birth.month 20 print(egg.course.period) 21 22 # 實例化方法二: 23 egg.birth=BirthDate(1996,22,4) 24 print(egg.birth.month)

1 class BirthDate: 2 def __init__(self,year,month,day): 3 self.year=year 4 self.month = month 5 self.day = day 6 class Course: 7 def __init__(self,name,price,period): #period為周期 8 self.name =name 9 self.price = price 10 self.period = period 11 class Teacher: 12 def __init__(self,name,salary,course): 13 self.name = name 14 self.salary = salary 15 self.course = course 16 # 17 # #在外面組合。(組合就是一個類的屬性是另一個類的對象) 18 19 egg = Teacher('egon',2000,'python') 20 egg.birth=BirthDate(1996,22,4) #直接給egg一個birth的屬性, 21 print(egg.birth.year) 22 23 egg.course =Course('python','6 months',15800) 24 print(egg.course.period)
二、繼承
1.繼承是一種創建新類的方式
2.新建的類可以創建一個或多個父類,父類有稱為基類或者超類
3.新建的類稱為派生類或者子類
在python中類的繼承分為:單繼承或多繼承
class ParentClass1: #定義父類 pass class ParentClass2: #定義父類 pass class SubClass1(ParentClass1): #單繼承,基類是ParentClass1,派生類是SubClass pass class SubClass2(ParentClass1,ParentClass2): #python支持多繼承,用逗號分隔開多個繼承的類 pass
4.查看所有繼承的父類
print(Person.__bases__) #__base __只查看從左到右繼承的第一個子類,__bases__則是查看所有繼承的父類
如果沒有指定父類,python會默認繼承object類,object是所有python的父類。
經典類:在python2中,class Dad: 不會繼承object,這樣的類叫做經典類(它叫經典類,不是因為它經典,而是因為它比較老)
新式類:在python3中,python會默認繼承object類(一切皆對象)
class Dad 就相當於python2中的 class Dad(object) #新式類
而且python3中沒有經典類了
5.繼承與抽象(先抽象后繼承)
抽象:抽取類似或者說比較像的部分(也就是提取一類事物的特點,范圍越來越大,共性越來越少)
是從大范圍到小范圍的過程
繼承:是基於抽象的過程,通過編程語言去實現它,肯定是先經歷抽象這個過程,才能通過繼承的方式去表達出抽象的結構
是從小范圍到大范圍的過程
6.派生:(相對論)
1.在父類的基礎上產生子類,產生的子類就叫做派生類
2.父類里沒有的方法,在子類中有了,這樣的方法就叫做派生方法。
3.父類里有,子類也有的方法,就叫做方法的重寫(就是把父類里的方法重寫了)
7.注意的幾個概念:
1.子類可以使用父類的所有屬性和方法
2.如果子類有自己的方法,就執行自己的;如果子類沒有自己的方法,就會找父類的。
3.如果子類里面沒有找到,父類里也沒有找到,就會報錯
4.如果子類中實現了調用父類的方法
在類內:super(子類,self).方法名() supper().__init__(參數)
在類外:super(子類名,對象名).方法名()
8.繼承的實例

1 class Animal: #父類 基類 超類 2 def __init__(self,name,life_value,aggr): 3 self.name= name 4 self.life_value = life_value 5 self.aggr = aggr #攻擊力 6 def eat(self): 7 self.life_value += 10 #誰調誰的血量就增加 8 9 class Person(Animal): #子類 派生類 10 def __init__(self, money, name, life_value, aggr): 11 super().__init__(name, life_value, aggr) 12 self.money = money #派生出來的一個屬性 13 def attack(self,enemy): #人的派生方法 14 #enemy.life_value = enemy.life_value - self.aggr 15 enemy.life_value -= self.aggr 16 17 class Dog(Animal): #子類 派生類 18 def __init__(self,name,breed, life_value,aggr): 19 # Animal.__init__(self,name,breed, life_value,aggr)#讓子類執行父類的方法 就是父類名.方法名(參數),連self都得傳 20 super().__init__(name,life_value,aggr) #super關鍵字 ,都不用傳self了,在新式類里的 21 # super(Dog,self).__init__(name,life_value,aggr) #上面super是簡寫 22 self.breed = breed 23 def bite(self,person): #狗的派生方法 24 person.life_value -= self.aggr 25 def eat(self): #父類方法的重寫 26 super().eat() 27 print('dog is eating') 28 29 # ha2 = Dog('旺財','哈士奇',20000,100) 30 # egg = Person('egon',500,1000,50) 31 # print(egg.aggr) 32 # print(ha2.aggr) 33 # egg.eat() 34 # print(egg.life_value) 35 # 36 # egg.eat() 37 # print(egg.life_value) 38 # 39 # ha2.eat() 40 # print(ha2.life_value) 41 # 42 # print(egg.life_value) 43 # ha2.bite(egg) 44 # print(egg.life_value) 45 # 46 47 ha2 = Dog('牛頭梗','旺財',20000,100) 48 print(ha2.life_value) 49 ha2.eat() #如果父類有的方法子類里面也有,那么就叫做方法的重寫,就不執行父類的了,去執行子類了 50 print(ha2.life_value) 51 52 53 super(Dog,ha2).eat() #生掉父類的方法,但是不推薦這樣用 54 print(ha2.life_value) 55 56 #在繼承中,如果子類有的方法就執行子類的 57 # 如果子類沒有的方法就執行父類的