面向?qū)ο缶幊?/h2>面向?qū)ο缶幊?Object Oriented Programming 簡稱 OOP,是一種程序設(shè)計思想。OOP把對象作為程序的基本單元,一個對象包含了數(shù)據(jù)和操作數(shù)據(jù)的函數(shù)。
而面向?qū)ο蟮某绦蛟O(shè)計把計算機程序視為一組對象的集合,而每個對象都可以接收其他對象發(fā)過來的消息,并處理這些消息,計算機程序的執(zhí)行就是一系列消息在各個對象之間傳遞。 在Python中,所有數(shù)據(jù)類型都可以視為對象,當(dāng)然也可以自定義對象。自定義的對象數(shù)據(jù)類型就是面向?qū)ο笾械念悾–lass)的概念。面向?qū)ο蟮某橄蟪潭扔直群瘮?shù)要高,因為一個Class既包含數(shù)據(jù),又包含操作數(shù)據(jù)的方法。 class Student(object): def __init__(self, name, score): self.name = name 類和實例class后面緊接著是類名,即Student,類名通常是大寫開頭的單詞,緊接著是(object),表示該類是從哪個類繼承下來的,繼承的概念我們后面再講,通常,如果沒有合適的繼承類,就使用object類,這是所有類最終都會繼承的類。 定義好了Student類,就可以根據(jù)Student類創(chuàng)建出Student的實例,創(chuàng)建實例是通過類名+()實現(xiàn)的。 數(shù)據(jù)封裝類本身擁有數(shù)據(jù)和方法,相當(dāng)于將“數(shù)據(jù)”封裝起來了。對于外部來說,并不需要知道內(nèi)部的邏輯。 訪問限制Class可以有屬性和方法,我們可以對屬性和方法進行控制,以達到允許或者不允許外部訪問的目的。如果要讓內(nèi)部屬性不被外部訪問,可以把屬性的名稱前加上兩個下劃線 class Student(object): def __init__(self, name, score): self.__name = name self.__score = score
繼承和多態(tài)在OOP程序設(shè)計中,當(dāng)我們定義一個class的時候,可以從某個現(xiàn)有的class繼承,新的class稱為子類(Subclass),而被繼承的class稱為基類、父類或超類(Base class、Super class)。 class Animal(object): def run(self): print('Animal is running…') 當(dāng)子類和父類都存在相同的run()方法時,我們說,子類的run()覆蓋了父類的run(),在代碼運行的時候,總是會調(diào)用子類的run()。這樣,我們就獲得了繼承的另一個好處:多態(tài)。在繼承關(guān)系中,如果一個實例的數(shù)據(jù)類型是某個子類,那它的數(shù)據(jù)類型也可以被看做是父類。但是,反過來就不行。 對于一個變量,我們只需要知道它是Animal類型,無需確切地知道它的子類型,就可以放心地調(diào)用run()方法,而具體調(diào)用的run()方法是作用在Animal、Dog、Cat還是Tortoise對象上,由運行時該對象的確切類型決定,這就是多態(tài)真正的威力:調(diào)用方只管調(diào)用,不管細節(jié),而當(dāng)我們新增一種Animal的子類時,只要確保run()方法編寫正確,不用管原來的代碼是如何調(diào)用的。這就是著名的“開閉”原則: 對擴展開放:允許新增Animal子類; 對修改封閉:不需要修改依賴Animal類型的外部函數(shù)。 鴨子類型對于靜態(tài)語言(例如Java)來說,如果需要傳入 獲取對象信息判斷Python中對象的類型,可以用以下方法。 type()基本類型都可以用 >>> import types isinstance()對于類和實例,使用 >>> isinstance([1, 2, 3], (list, tuple))True dir()如果要獲得一個對象的所有屬性和方法,可以使用 實例屬性和類屬性Python類創(chuàng)建的實例可以任意綁定屬性,如果需要對類本身綁定屬性,則需要在類中定義,這就區(qū)分了類屬性和實例屬性。
面向?qū)ο蟾呒壘幊?/h2>數(shù)據(jù)封裝、繼承和多態(tài)是面相對象程序設(shè)計中的三個基本概念,另外還有很多特性,包括多重繼承、定制類等。 使用 slots()在Python中,可以對類動態(tài)的增加屬性和方法,這在靜態(tài)語言中很難實現(xiàn)。 #!/usr/bin/env python3# -*- coding: utf-8 -*- 但這也帶來一個問題,屬性和方法可以隨意更改,如果我們要限制怎么辦?可以使用 class Student(object): __slots__ = ('name', 'age') # 用tuple定義允許綁定的屬性名稱 使用@PropertyPython中實例的屬性暴露在外面可以隨便修改,這樣就無法保證屬性的有效性符合校驗規(guī)則。雖然可以通過設(shè)置Setter和Getter來進行檢查,但如果屬性特別多,操作起來又比較麻煩。 還記得裝飾器(decorator)可以給函數(shù)動態(tài)加上功能嗎?對于類的方法,裝飾器一樣起作用。Python內(nèi)置的@property裝飾器就是負責(zé)把一個方法變成屬性調(diào)用的。 class Student(object): @property def score(self): return self._score @score.setter def score(self, value): if not isinstance(value, int):
>>> s = Student() 還可以定義只讀屬性,只定義getter方法,不定義setter方法就是一個只讀屬性。 class Student(object): @property def birth(self): return self._birth @birth.setter def birth(self, value): self._birth = value @property def age(self): return 2015 - self._birth 多重繼承繼承是面向?qū)ο缶幊痰囊粋€重要的方式,因為通過繼承,子類就可以擴展父類的功能。舉例來說對于動物的對象設(shè)計,可以按照“哺乳動物”、“鳥類”來設(shè)計分類對象,按照不同的維度,也可以按照“能跑的”、“能飛的”或者“寵物”、“非寵物”設(shè)計分類,如果按照單一繼承的方式,類的設(shè)計就像下圖,會變的非常復(fù)雜。 正確的辦法是采用多重繼承。一個子類就可以同時獲得多個父類的所有功能。在設(shè)計類的繼承關(guān)系時,通常,主線都是單一繼承下來的,例如,Ostrich繼承自Bird。但是,如果需要“混入”額外的功能,通過多重繼承就可以實現(xiàn),比如,讓Ostrich除了繼承自Bird外,再同時繼承Runnable。這種設(shè)計通常稱之為MixIn。 class Animal(object): pass# 大類:class Mammal(Animal): passclass Bird(Animal): pass# 各種動物:class Dog(Mammal): passclass Bat(Mammal): passclass Parrot(Bird): passclass Ostrich(Bird): passclass Runnable(object): def run(self): print('Running...')class Flyable(object): def fly(self): print('Flying…’)class Dog(Mammal, Runnable): passclass Bat(Mammal, Flyable): pass Python自帶的很多庫也使用了MixIn。舉個例子,Python自帶了TCPServer和UDPServer這兩類網(wǎng)絡(luò)服務(wù),而要同時服務(wù)多個用戶就必須使用多進程或多線程模型,這兩種模型由ForkingMixIn和ThreadingMixIn提供。通過組合,我們就可以創(chuàng)造出合適的服務(wù)來。 定制類類似于 str定義 >>> class Student(object):... repr定義返回程序開發(fā)者看到的字符串,也就是在命令行狀態(tài)下執(zhí)行時的返回值。 class Student(object): def __init__(self, name): self.name = name iter如果一個類想被用于 class Fib(object): def __init__(self): self.a, self.b = 0, 1 # 初始化兩個計數(shù)器a,b def __iter__(self): return self # 實例本身就是迭代對象,故返回自己 def __next__(self): self.a, self.b = self.b, self.a + self.b # 計算下一個值 if self.a > 100000: # 退出循環(huán)的條件 raise StopIteration() getitemFib實例雖然能作用于for循環(huán),看起來和list有點像,但是,把它當(dāng)成list來使用還是不行,比如,取第5個元素。 class Fib(object): def __getitem__(self, n): if isinstance(n, int): # n是索引 a, b = 1, 1 for x in range(n): a, b = b, a + b 通過上面的方法,我們自己定義的類表現(xiàn)得和Python自帶的list、tuple、dict沒什么區(qū)別,這完全歸功于動態(tài)語言的“鴨子類型”,不需要強制繼承某個接口。 getattr正常情況下,當(dāng)我們調(diào)用類的方法或?qū)傩詴r,如果不存在,就會報錯。要避免這個錯誤,除了可以加上一個屬性外,Python還有另一個機制,那就是寫一個__getattr__()方法,動態(tài)返回一個屬性。 class Student(object): def __init__(self): self.name = 'Michael' def __getattr__(self, attr): if attr=='score': call任何類,只需要定義一個 class Student(object): def __init__(self, name): self.name = name 使用枚舉類之前說到過,Python中其實不存在常量,但是可以通過枚舉類的方式來變通實現(xiàn)。 from enum import EnumMonth = Enum('Month', ('Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec')) 這樣我們就獲得了Month類型的枚舉類,可以直接使用Month.Jan來引用一個常量,或者枚舉它的所有成員。 使用元類type()type()函數(shù)可以查看一個類型或變量的類型,Hello是一個class,它的類型就是type,而h是一個實例,它的類型就是class Hello。我們說class的定義是運行時動態(tài)創(chuàng)建的,而創(chuàng)建class的方法就是使用type()函數(shù)。 type()函數(shù)既可以返回一個對象的類型,又可以創(chuàng)建出新的類型,比如,我們可以通過type()函數(shù)創(chuàng)建出Hello類,而無需通過class Hello(object)…的定義。 >>> def fn(self, name='world'): # 先定義函數(shù)... 要創(chuàng)建一個class對象,type()函數(shù)依次傳入3個參數(shù): 1.class的名稱; 通過type()函數(shù)創(chuàng)建的類和直接寫class是完全一樣的,因為Python解釋器遇到class定義時,僅僅是掃描一下class定義的語法,然后調(diào)用type()函數(shù)創(chuàng)建出class。 metaclassmetaclass,直譯為元類,簡單的解釋就是: # metaclass是類的模板,所以必須從`type`類型派生: 感謝大家對“Python互動中心”的關(guān)注,一起學(xué)習(xí)的可加小編為好友 |
|