No.1 一切皆對(duì)象
眾所周知,Java中強(qiáng)調(diào)“一切皆對(duì)象”,但是Python中的面向?qū)ο蟊菾ava更加徹底,因?yàn)镻ython中的類(class)也是對(duì)象,函數(shù)(function)也是對(duì)象,而且Python的代碼和模塊也都是對(duì)象。
Python中函數(shù)和類可以賦值給一個(gè)變量
Python中函數(shù)和類可以存放到集合對(duì)象中
Python中函數(shù)和類可以作為一個(gè)函數(shù)的參數(shù)傳遞給函數(shù)
Python中函數(shù)和類可以作為返回值
Step.1
# 首先創(chuàng)建一個(gè)函數(shù)和一個(gè)Python3.x的新式類class Demo(object): def __init__(self): print(“Demo Class”)# 定義一個(gè)函數(shù)def function(): print(“function”)# 在Python無論是函數(shù),還是類,都是對(duì)象,他們可以賦值給一個(gè)變量class_value = Demofunc_value = function# 并且可以通過變量調(diào)用class_value() # Demo Classfunc_value() # function
Step.2
‘’‘遇到問題沒人解答?小編創(chuàng)建了一個(gè)Python學(xué)習(xí)交流QQ群:857662006 尋找有志同道合的小伙伴,互幫互助,群里還有不錯(cuò)的視頻學(xué)習(xí)教程和PDF電子書!’‘’# 將函數(shù)和類添加到集合中obj_list = []obj_list.append(Demo)obj_list.append(function)# 遍歷列表for i in obj_list: print(i) # 《class ‘__main__.Demo’》 # 《function function at 0x0000020D681B3E18》
Step.3
# 定義一個(gè)具體函數(shù)def test_func(class_name, func_name): class_name() func_name()# 將類名和函數(shù)名傳入形參列表test_func(Demo, function)# Demo Class# function
Step.4
# 定義函數(shù)實(shí)現(xiàn)返回類和函數(shù)def test_func2(): return Demodef test_func3(): return function# 執(zhí)行函數(shù)test_func2()() # Demo Classtest_func3()() # function
No.2 關(guān)鍵字type、object、class之間的關(guān)系
在Python中,object的實(shí)例是type,object是頂層類,沒有基類;type的實(shí)例是type,type的基類是object。Python中的內(nèi)置類型的基類是object,但是他們都是由type實(shí)例化而來,具體的值由內(nèi)置類型實(shí)例化而來。在Python2.x的語法中用戶自定義的類沒有明確指定基類就默認(rèn)是沒有基類,在Python3.x的語法中,指定基類為object。
‘’‘遇到問題沒人解答?小編創(chuàng)建了一個(gè)Python學(xué)習(xí)交流QQ群:857662006 尋找有志同道合的小伙伴,互幫互助,群里還有不錯(cuò)的視頻學(xué)習(xí)教程和PDF電子書!’‘’# object是誰實(shí)例化的?print(type(object)) # 《class ‘type’》# object繼承自哪個(gè)類?print(object.__bases__) # ()# type是誰實(shí)例化的?print(type(type)) # 《class ‘type’》# type繼承自哪個(gè)類?print(type.__bases__) # (《class ‘object’》,)# 定義一個(gè)變量value = 100# 100由誰實(shí)例化?print(type(value)) # 《class ‘int’》# int由誰實(shí)例化?print(type(int)) # 《class ‘type’》# int繼承自哪個(gè)類?print(int.__bases__) # (《class ‘object’》,)# Python 2.x的舊式類class OldClass(): pass# Python 3.x的新式類class NewClass(object): pass
No.3 Python的內(nèi)置類型
在Python中,對(duì)象有3個(gè)特征屬性:
在內(nèi)存中的地址,使用id()函數(shù)進(jìn)行查看
對(duì)象的類型
對(duì)象的默認(rèn)值
Step.1 None類型
在Python解釋器啟動(dòng)時(shí),會(huì)創(chuàng)建一個(gè)None類型的None對(duì)象,并且None對(duì)象全局只有一個(gè)。
Step.2 數(shù)值類型
ini類型
float類型
complex類型
bool類型
Step.3 迭代類型
在Python中,迭代類型可以使用循環(huán)來進(jìn)行遍歷。
Step.4 序列類型
list
tuple
str
array
range
bytes, bytearray, memoryvie(二進(jìn)制序列)
Step.5 映射類型
dict
Step.6 集合類型
set
frozenset
Step.7 上下文管理類型
with語句
Step.8 其他類型
模塊
class
實(shí)例
函數(shù)
方法
代碼
object對(duì)象
type對(duì)象
ellipsis(省略號(hào))
notimplemented
NO.4 魔法函數(shù)
Python中的魔法函數(shù)使用雙下劃線開始,以雙下劃線結(jié)尾。關(guān)于詳細(xì)介紹請(qǐng)看我的文章——《全面總結(jié)Python中的魔法函數(shù)》。
No.5 鴨子類型與白鵝類型
鴨子類型是程序設(shè)計(jì)中的推斷風(fēng)格,在鴨子類型中關(guān)注對(duì)象如何使用而不是類型本身。鴨子類型像多態(tài)一樣工作但是沒有繼承。鴨子類型的概念來自于:“當(dāng)看到一只鳥走起來像鴨子、游泳起來像鴨子、叫起來也像鴨子,那么這只鳥就可以被稱為鴨子?!?/p>
‘’‘遇到問題沒人解答?小編創(chuàng)建了一個(gè)Python學(xué)習(xí)交流QQ群:857662006 尋找有志同道合的小伙伴,互幫互助,群里還有不錯(cuò)的視頻學(xué)習(xí)教程和PDF電子書!’‘’# 定義狗類class Dog(object): def eat(self): print(“dog is eatting.。.”)# 定義貓類class Cat(object): def eat(self): print(“cat is eatting.。.”)# 定義鴨子類class Duck(object): def eat(self): print(“duck is eatting.。.”)# 以上Python中多態(tài)的體現(xiàn)# 定義動(dòng)物列表an_li = []# 將動(dòng)物添加到列表an_li.append(Dog)an_li.append(Cat)an_li.append(Duck)# 依次調(diào)用每個(gè)動(dòng)物的eat()方法for i in an_li: i().eat()# dog is eatting.。.# cat is eatting.。.# duck is eatting.。.
白鵝類型是指只要 cls 是抽象基類,即 cls 的元類是 abc.ABCMeta ,就可以使用 isinstance(obj, cls)。
No.6 協(xié)議、 抽象基類、abc模塊和序列之間的繼承關(guān)系
協(xié)議:Python中的非正式接口,是允許Python實(shí)現(xiàn)多態(tài)的方式,協(xié)議是非正式的,不具備強(qiáng)制性,由約定和文檔定義。
接口:泛指實(shí)體把自己提供給外界的一種抽象化物(可以為另一實(shí)體),用以由內(nèi)部操作分離出外部溝通方法,使其能被內(nèi)部修改而不影響外界其他實(shí)體與其交互的方式。
我們可以使用猴子補(bǔ)丁來實(shí)現(xiàn)協(xié)議,那么什么是猴子補(bǔ)丁呢?
猴子補(bǔ)丁就是在運(yùn)行時(shí)修改模塊或類,不去修改源代碼,從而實(shí)現(xiàn)目標(biāo)協(xié)議接口操作,這就是所謂的打猴子補(bǔ)丁。
Tips:猴子補(bǔ)丁的叫法起源于Zope框架,開發(fā)人員在修改Zope的Bug時(shí),經(jīng)常在程序后面追加更新的部分,這些雜牌軍補(bǔ)丁的英文名字叫做guerilla patch,后來寫成gorllia,接著就變成了monkey。
猴子補(bǔ)丁的主要作用是:
在運(yùn)行時(shí)替換方法、屬性
在不修改源代碼的情況下對(duì)程序本身添加之前沒有的功能
在運(yùn)行時(shí)對(duì)象中添加補(bǔ)丁,而不是在磁盤中的源代碼上
應(yīng)用案例:假設(shè)寫了一個(gè)很大的項(xiàng)目,處處使用了json模塊來解析json文件,但是后來發(fā)現(xiàn)ujson比json性能更高,修改源代碼是要修改很多處的,所以只需要在程序入口加入:
import json# pip install ujsonimport ujson def monkey_patch_json(): json.__name__ = ‘ujson’ json.dumps = ujson.dumps json.loads = ujson.loads monkey_patch_json()
Python 的抽象基類有一個(gè)重要實(shí)用優(yōu)勢(shì):可以使用 register 類方法在終端用戶的代碼中把某個(gè)類 “聲明” 為一個(gè)抽象基類的 “虛擬” 子 類(為此,被注冊(cè)的類必腨滿足抽象其類對(duì)方法名稱和簽名的要求,最重要的是要滿足底 層語義契約;但是,開發(fā)那個(gè)類時(shí)不用了解抽象基類,更不用繼承抽象基類 。有時(shí),為了讓抽象類識(shí)別子類,甚至不用注冊(cè)。要抑制住創(chuàng)建抽象基類的沖動(dòng)。濫用抽象基類會(huì)造成災(zāi)難性后果,表明語言太注重表面形式 。
抽象基類不能被實(shí)例化(不能創(chuàng)建對(duì)象),通常是作為基類供子類繼承,子類中重寫虛函數(shù),實(shí)現(xiàn)具體的接口。
判定某個(gè)對(duì)象的類型
強(qiáng)制子類必須實(shí)現(xiàn)某些方法
抽象基類的定義與使用
import abc# 定義緩存類class Cache(metaclass=abc.ABCMeta): @abc.abstractmethod def get(self, key): pass @abc.abstractmethod def set(self, key, value): pass# 定義redis緩存類實(shí)現(xiàn)Cache類中的get()和set()方法class RedisCache(Cache): def set(self, key): pass def get(self, key, value): pass
值得注意的是:Python 3.0-Python3.3之間,繼承抽象基類的語法是class ClassName(metaclass=adc.ABCMeta),其他版本是:class ClassName(abc.ABC)。
collections.abc模塊中各個(gè)抽象基類的UML類圖
35個(gè)高級(jí)Python知識(shí)點(diǎn)總結(jié)
No.7 isinstence和type的區(qū)別
class A(object): passclass B(A): passb = B()print(isinstance(b, B))print(isinstance(b, A))print(type(b) is B)print(type(b) is A)# True# True# True# False
No.8 類變量和實(shí)例變量
實(shí)例變量只能通過類的實(shí)例進(jìn)行調(diào)用
修改模板對(duì)象創(chuàng)建的對(duì)象的屬性,模板對(duì)象的屬性不會(huì)改變
修改模板對(duì)象的屬性,由模板對(duì)象創(chuàng)建的對(duì)象的屬性會(huì)改變
‘’‘遇到問題沒人解答?小編創(chuàng)建了一個(gè)Python學(xué)習(xí)交流QQ群:857662006 尋找有志同道合的小伙伴,互幫互助,群里還有不錯(cuò)的視頻學(xué)習(xí)教程和PDF電子書!’‘’# 此處的類也是模板對(duì)象,Python中一切皆對(duì)象class A(object): #類變量 number = 12 def __init__(self): # 實(shí)例變量 self.number_2 = 13# 實(shí)例變量只能通過類的實(shí)例進(jìn)行調(diào)用print(A.number) # 12print(A().number) # 12print(A().number_2) # 13# 修改模板對(duì)象創(chuàng)建的對(duì)象的屬性,模板對(duì)象的屬性不會(huì)改變a = A()a.number = 18print(a.number) # 18print(A().number) # 12print(A.number) # 12# 修改模板對(duì)象的屬性,由模板對(duì)象創(chuàng)建的對(duì)象的屬性會(huì)改變A.number = 19print(A.number) # 19print(A().number) # 19
No.9 類和實(shí)例屬性以及方法的查找順序
在Python 2.2之前只有經(jīng)典類,到Python2.7還會(huì)兼容經(jīng)典類,Python3.x以后只使用新式類,Python之前版本也會(huì)兼容新式類
Python 2.2 及其之前類沒有基類,Python新式類需要顯式繼承自object,即使不顯式繼承也會(huì)默認(rèn)繼承自object
經(jīng)典類在類多重繼承的時(shí)候是采用從左到右深度優(yōu)先原則匹配方法的。而新式類是采用C3算法
經(jīng)典類沒有MRO和instance.mro()調(diào)用的
假定存在以下繼承關(guān)系:
class D(object): def say_hello(self): passclass E(object): passclass B(D): passclass C(E): passclass A(B, C): pass
采用DFS(深度優(yōu)先搜索算法)當(dāng)調(diào)用了A的say_hello()方法的時(shí)候,系統(tǒng)會(huì)去B中查找如果B中也沒有找到,那么去D中查找,很顯然D中存在這個(gè)方法,但是DFS對(duì)于以下繼承關(guān)系就會(huì)有缺陷:
class D(object): passclass B(D): passclass C(D): def say_hello(self): passclass A(B, C): pass
在A的實(shí)例對(duì)象中調(diào)用say_hello方法時(shí),系統(tǒng)會(huì)先去B中查找,由于B類中沒有該方法的定義,所以會(huì)去D中查找,D類中也沒有,系統(tǒng)就會(huì)認(rèn)為該方法沒有定義,其實(shí)該方法在C中定義了。所以考慮使用BFS(廣度優(yōu)先搜索算法),那么問題回到第一個(gè)繼承關(guān)系,假定C和D具備重名方法,在調(diào)用A的實(shí)例的方法時(shí),應(yīng)該先在B中查找,理應(yīng)調(diào)用D中的方法,但是使用BFS的時(shí)候,C類中的方法會(huì)覆蓋D類中的方法。在Python 2.3以后的版本中,使用C3算法:
# 獲取解析順序的方法類名.mro()類名.__mro__inspect.getmro(類名)
使用C3算法后的第二種繼承順序:
‘’‘遇到問題沒人解答?小編創(chuàng)建了一個(gè)Python學(xué)習(xí)交流QQ群:857662006 尋找有志同道合的小伙伴,互幫互助,群里還有不錯(cuò)的視頻學(xué)習(xí)教程和PDF電子書!’‘’class D(object): passclass B(D): passclass C(D): def say_hello(self): passclass A(B, C): passprint(A.mro()) # [《class ‘__main__.A’》, 《class ‘__main__.B’》, 《class ‘__main__.C’》, 《class ‘__main__.D’》, 《class ‘object’》]
使用C3算法后的第一種繼承順序:
class D(object): passclass E(object): passclass B(D): passclass C(E): passclass A(B, C): passprint(A.mro()) # [《class ‘__main__.A’》, 《class ‘__main__.B’》, 《class ‘__main__.D’》, 《class ‘__main__.C’》, 《class ‘__main__.E’》, 《class ‘object’》]
在這里僅介紹了算法的作用和演變歷史,關(guān)于深入詳細(xì)解析,請(qǐng)看我的其他文章——《從Python繼承談起,到C3算法落筆》。
No.10 類方法、實(shí)例方法和靜態(tài)方法
class Demo(object): # 類方法 @classmethod def class_method(cls, number): pass # 靜態(tài)方法 @staticmethod def static_method(number): pass # 對(duì)象方法/實(shí)例方法 def object_method(self, number): pass
實(shí)例方法只能通過類的實(shí)例來調(diào)用;靜態(tài)方法是一個(gè)獨(dú)立的、無狀態(tài)的函數(shù),緊緊依托于所在類的命名空間上;類方法在為了獲取類中維護(hù)的數(shù)據(jù),比如:
class Home(object): # 房間中人數(shù) __number = 0 @classmethod def add_person_number(cls): cls.__number += 1 @classmethod def get_person_number(cls): return cls.__number def __new__(self): Home.add_person_number() # 重寫__new__方法,調(diào)用object的__new__ return super().__new__(self)class Person(Home): def __init__(self): # 房間人員姓名 self.name = ‘name’ # 創(chuàng)建人員對(duì)象時(shí)調(diào)用Home的__new__()方法tom = Person()print(type(tom)) # 《class ‘__main__.Person’》alice = Person()bob = Person()test = Person()print(Home.get_person_number())
-
JAVA
+關(guān)注
關(guān)注
20文章
3012瀏覽量
116874 -
函數(shù)
+關(guān)注
關(guān)注
3文章
4422瀏覽量
67871 -
python
+關(guān)注
關(guān)注
58文章
4889瀏覽量
90330
發(fā)布評(píng)論請(qǐng)先 登錄
關(guān)于車速傳感器磁鐵的一些知識(shí)點(diǎn)
MOSFET管開關(guān)電路基本知識(shí)總結(jié)
示波器小知識(shí)點(diǎn)
【黑金云課堂筆記】第一~二期知識(shí)點(diǎn)總結(jié)
【黑金云課堂筆記】第三期知識(shí)點(diǎn)總結(jié)
Python的知識(shí)點(diǎn)總結(jié)說明
評(píng)論