繼承和組合


一、組合

組合:組合指的是,在一個類中以另外一個類的對象(也就是實例)作為數據屬性,稱為類的組合

   也就是說:一個類的屬性是另一個類的對象,就是組合

例子:

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

 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)
在__init__里面組合
 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 # 如果子類沒有的方法就執行父類的
實現調用父類的方法

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM