1. 程式人生 > >python全棧開發【第十四篇】面向對象三大特性——繼承

python全棧開發【第十四篇】面向對象三大特性——繼承

blog area ear person subclass rom from 指定 實現

一、組合

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

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

例子:

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

#求圓環的面積和周長
from math import pi
class Circle:
    def __init__(self,r):
        self.r=r
    def perimater(self):
        return 2*pi*self.r
    def area(self):
        return pi*self.r*self.r
# print(Circle.perimater(‘r‘,2))
# print(Circle.area(‘r‘,3))


class Circle_ring: #定義一個圓環類
    def __init__(self,outside_r,inside_r):
        outside_bijiao = max(outside_r,inside_r)
        intside_bijiao = min(outside_r, inside_r)
        self.outsize_circle = Circle(outside_bijiao) #實例化一個大圓形  作為self.outside_circle屬性的值
        self.intsize_circle = Circle(intside_bijiao) #實例化一個小圓環
    def area(self):
        return self.outsize_circle.area()-self.intsize_circle.area()
    def perimater(self):
        return self.intsize_circle.perimater()+self.outsize_circle.perimater()


r1 = Circle_ring(10,20)  #實例化
print(r1.area())
print(r1.perimater())
組合的兩種方式:1.在__init__方法裏面組合
2.在外面組合
#在__init__裏面組合
class BirthDate:
    def __init__(self,year,month,day):
        self.year=year
        self.month = month
        self.day = day
class Course:
    def __init__(self,name,price,period): #period為周期
        self.name =name
        self.price = price
        self.period = period
class Teacher:
    def __init__(self,name,salary,year,month,day,price,period): #那麽這個裏面也要把該有的屬性傳進去
        self.birth = BirthDate(year,month,day) #在裏面組合(將BirthDate裏面有的屬性傳入進去)
        self.course=Course(name,price,period)
        self.name = name
        self.salary = salary
# 實例化方法一:
egg = Teacher(‘egon‘,2000,1999,12,2,‘6 months‘,‘15800‘)  #也要實例化,Teacher類裏面的屬性都得實例化
print(egg.birth.month)  #當然老師也有生日,就讓egg.birth.month
print(egg.course.period)

# 實例化方法二:
egg.birth=BirthDate(1996,22,4)
print(egg.birth.month)
#在類外面實例化組合
class BirthDate:
    def __init__(self,year,month,day):
        self.year=year
        self.month = month
        self.day = day
class Course:
    def __init__(self,name,price,period): #period為周期
        self.name =name
        self.price = price
        self.period = period
class Teacher:
    def __init__(self,name,salary,course):
        self.name = name
        self.salary = salary
        self.course = course
#
# #在外面組合。(組合就是一個類的屬性是另一個類的對象)

egg = Teacher(‘egon‘,2000,‘python‘)
egg.birth=BirthDate(1996,22,4) #直接給egg一個birth的屬性,
print(egg.birth.year)

egg.course =Course(‘python‘,‘6 months‘,15800)
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.繼承的實例

#實現調用父類的方法
class Animal:      #父類  基類  超類
    def __init__(self,name,life_value,aggr):
        self.name= name
        self.life_value = life_value
        self.aggr = aggr  #攻擊力
    def eat(self):
        self.life_value += 10  #誰調誰的血量就增加

class Person(Animal):  #子類  派生類
    def __init__(self, money, name, life_value, aggr):
        super().__init__(name, life_value, aggr)
        self.money = money  #派生出來的一個屬性
    def attack(self,enemy):    #人的派生方法
        #enemy.life_value = enemy.life_value - self.aggr
        enemy.life_value -= self.aggr

class Dog(Animal): #子類  派生類
    def __init__(self,name,breed, life_value,aggr):
        # Animal.__init__(self,name,breed, life_value,aggr)#讓子類執行父類的方法 就是父類名.方法名(參數),連self都得傳
        super().__init__(name,life_value,aggr) #super關鍵字  ,都不用傳self了,在新式類裏的
        # super(Dog,self).__init__(name,life_value,aggr)  #上面super是簡寫
        self.breed = breed
    def bite(self,person):   #狗的派生方法
        person.life_value -= self.aggr
    def eat(self):  #父類方法的重寫
        super().eat()
        print(‘dog is eating‘)

# ha2 = Dog(‘旺財‘,‘哈士奇‘,20000,100)
# egg = Person(‘egon‘,500,1000,50)
# print(egg.aggr)
# print(ha2.aggr)
# egg.eat()
# print(egg.life_value)
#
# egg.eat()
# print(egg.life_value)
#
# ha2.eat()
# print(ha2.life_value)
#
# print(egg.life_value)
# ha2.bite(egg)
# print(egg.life_value)
#

ha2 = Dog(‘牛頭梗‘,‘旺財‘,20000,100)
print(ha2.life_value)
ha2.eat()  #如果父類有的方法子類裏面也有,那麽就叫做方法的重寫,就不執行父類的了,去執行子類了
print(ha2.life_value)


super(Dog,ha2).eat() #生掉父類的方法,但是不推薦這樣用
print(ha2.life_value)

#在繼承中,如果子類有的方法就執行子類的
# 如果子類沒有的方法就執行父類的

  

python全棧開發【第十四篇】面向對象三大特性——繼承