魔法函數(shù)python 魔法函數(shù)Python為啥報(bào)錯(cuò)

Python中常見(jiàn)魔法方法介紹

魔法方法 (Magic Methods) 是Python中的內(nèi)置函數(shù),一般以雙下劃線(xiàn)開(kāi)頭和結(jié)尾,例如__ init__ 、 __del__ 等。之所以稱(chēng)之為魔法方法,是因?yàn)檫@些方法會(huì)在進(jìn)行特定的操作時(shí)會(huì)自動(dòng)被調(diào)用。

創(chuàng)新互聯(lián)是一家集網(wǎng)站建設(shè),泰來(lái)企業(yè)網(wǎng)站建設(shè),泰來(lái)品牌網(wǎng)站建設(shè),網(wǎng)站定制,泰來(lái)網(wǎng)站建設(shè)報(bào)價(jià),網(wǎng)絡(luò)營(yíng)銷(xiāo),網(wǎng)絡(luò)優(yōu)化,泰來(lái)網(wǎng)站推廣為一體的創(chuàng)新建站企業(yè),幫助傳統(tǒng)企業(yè)提升企業(yè)形象加強(qiáng)企業(yè)競(jìng)爭(zhēng)力??沙浞譂M(mǎn)足這一群體相比中小企業(yè)更為豐富、高端、多元的互聯(lián)網(wǎng)需求。同時(shí)我們時(shí)刻保持專(zhuān)業(yè)、時(shí)尚、前沿,時(shí)刻以成就客戶(hù)成長(zhǎng)自我,堅(jiān)持不斷學(xué)習(xí)、思考、沉淀、凈化自己,讓我們?yōu)楦嗟钠髽I(yè)打造出實(shí)用型網(wǎng)站。

在Python中,可以通過(guò)dir()方法來(lái)查看某個(gè)對(duì)象的所有方法和屬性,其中雙下劃線(xiàn)開(kāi)頭和結(jié)尾的就是該對(duì)象的魔法方法。以字符串對(duì)象為例:

可以看到字符串對(duì)象有 __add__ 方法,所以在Python中可以直接對(duì)字符串對(duì)象使用"+"操作,當(dāng)Python識(shí)別到"+"操作時(shí),就會(huì)調(diào)用該對(duì)象的 __add__ 方法。有需要時(shí)我們可以在自己的類(lèi)中重寫(xiě) __add__ 方法來(lái)完成自己想要的效果。

我們重寫(xiě)了 __add__ 方法,當(dāng)Python識(shí)別"+"操作時(shí),會(huì)自動(dòng)調(diào)用重寫(xiě)后的 __add__ 方法??梢钥吹?,魔法方法在類(lèi)或?qū)ο蟮哪承┦录霭l(fā)后會(huì)自動(dòng)執(zhí)行,如果希望根據(jù)自己的程序定制特殊功能的類(lèi),那么就需要對(duì)這些方法進(jìn)行重寫(xiě)。使用魔法方法,我們可以非常方便地給類(lèi)添加特殊的功能。

1、構(gòu)造與初始化

__ new __ 、 __ init __ 這兩個(gè)魔法方法常用于對(duì)類(lèi)的初始化操作。上面我們創(chuàng)建a1 = A("hello")時(shí),但首先調(diào)用的是 __ new __ ;初始化一個(gè)類(lèi)分為兩步:

a.調(diào)用該類(lèi)的new方法,返回該類(lèi)的實(shí)例對(duì)象

b.調(diào)用該類(lèi)的init方法,對(duì)實(shí)例對(duì)象進(jìn)行初始化。

__new__ (cls, *args, **kwargs)至少需要一個(gè)cls參數(shù),代表傳入的類(lèi)。后面兩個(gè)參數(shù)傳遞給 __ init __ 。在 __ new __ 可以決定是否繼續(xù)調(diào)用 __ init __ 方法,只有當(dāng) __ new __ 返回了當(dāng)前類(lèi)cls的實(shí)例,才會(huì)接著調(diào)用 __ init __ 。結(jié)合 __ new __ 方法的特性,我們可以通過(guò)重寫(xiě) __ new __ 方法實(shí)現(xiàn)Python的單例模式:

可以看到雖然創(chuàng)建了兩個(gè)對(duì)象,但兩個(gè)對(duì)象的地址相同。

2、控制屬性訪(fǎng)問(wèn)這類(lèi)魔法

方法主要對(duì)對(duì)象的屬性進(jìn)行訪(fǎng)問(wèn)、定義、修改時(shí)起作用。主要有:

__getattr__(self, name): 定義當(dāng)用戶(hù)試圖獲取一個(gè)屬性時(shí)的行為。

__getattribute__(self, name):定義當(dāng)該類(lèi)的屬性被訪(fǎng)問(wèn)時(shí)的行為(先調(diào)用該方法,查看是否存在該屬性,若不存在,接著去調(diào)用getattr)。

__setattr__(self, name, value):定義當(dāng)一個(gè)屬性被設(shè)置時(shí)的行為。

當(dāng)初始化屬性時(shí)如self.a=a時(shí)或修改實(shí)例屬性如ins.a=1時(shí)本質(zhì)時(shí)調(diào)用魔法方法self. __ setattr __ (name,values);當(dāng)實(shí)例訪(fǎng)問(wèn)某個(gè)屬性如ins.a本質(zhì)是調(diào)用魔法方法a. __ getattr __ (name)

3、容器類(lèi)操作

有一些方法可以讓我們自己定義自己的容器,就像Python內(nèi)置的List,Tuple,Dict等等;容器分為可變?nèi)萜骱筒豢勺內(nèi)萜鳌?/p>

如果自定義一個(gè)不可變?nèi)萜鞯脑?huà),只能定義__ len__ 和__ getitem__ ;定義一個(gè)可變?nèi)萜鞒瞬豢勺內(nèi)萜鞯乃心Хǚ椒?,還需要定義__ setitem__ 和__ delitem__ ;如果容器可迭代。還需要定義__ iter __。

__len__(self):返回容器的長(zhǎng)度

__getitem__(self,key):當(dāng)需要執(zhí)行self[key]的方式去調(diào)用容器中的對(duì)象,調(diào)用的是該方法

__setitem__(self,key,value):當(dāng)需要執(zhí)行self[key] = value時(shí),調(diào)用的是該方法

__iter__(self):當(dāng)容器可以執(zhí)行 for x in container:,或者使用iter(container)時(shí),需要定義該方法

下面舉一個(gè)例子,實(shí)現(xiàn)一個(gè)容器,該容器有List的一般功能,同時(shí)增加一些其它功能如訪(fǎng)問(wèn)第一個(gè)元素,最后一個(gè)元素,記錄每個(gè)元素被訪(fǎng)問(wèn)的次數(shù)等。

這類(lèi)方法的使用場(chǎng)景主要在你需要定義一個(gè)滿(mǎn)足需求的容器類(lèi)數(shù)據(jù)結(jié)構(gòu)時(shí)會(huì)用到,比如可以嘗試自定義實(shí)現(xiàn)樹(shù)結(jié)構(gòu)、鏈表等數(shù)據(jù)結(jié)構(gòu)(在collections中均已有),或者項(xiàng)目中需要定制的一些容器類(lèi)型。

魔法方法在Python代碼中能夠簡(jiǎn)化代碼,提高代碼可讀性,在常見(jiàn)的Python第三方庫(kù)中可以看到很多對(duì)于魔法方法的運(yùn)用。

因此當(dāng)前這篇文章僅是拋磚引玉,真正的使用需要在開(kāi)源的優(yōu)秀源碼中以及自身的工程實(shí)踐中不斷加深理解并合適應(yīng)用。

_是什么意思

__init__屬于魔法函數(shù)的一種,讓我們來(lái)看一下它的前世今生吧。后面介紹了其他的魔法函數(shù):__ str__()、__ new__()、__ unicode__()、__ call__()、__ len__()、__repr__()等等

所謂魔法函數(shù)(Magic Methods),是Python的一種高級(jí)語(yǔ)法,允許你在類(lèi)中自定義函數(shù)(函數(shù)名格式一般為_(kāi)_xx__),并綁定到類(lèi)的特殊方法中。比如在類(lèi)A中自定義__str__()函數(shù),則在調(diào)用str(A())時(shí),會(huì)自動(dòng)調(diào)用__str__()函數(shù),并返回相應(yīng)的結(jié)果。在我們平時(shí)的使用中,可能經(jīng)常使用__init__函數(shù)(構(gòu)造函數(shù))和__del__函數(shù)(析構(gòu)函數(shù)),其實(shí)這也是魔法函數(shù)的一種。

Python中以雙下劃線(xiàn)(__xx__)開(kāi)始和結(jié)束的函數(shù)(不可自己定義)為魔法函數(shù)。

調(diào)用類(lèi)實(shí)例化的對(duì)象的方法時(shí)自動(dòng)調(diào)用魔法函數(shù)。

在自己定義的類(lèi)中,可以實(shí)現(xiàn)之前的內(nèi)置函數(shù)。

Python魔法函數(shù)(特殊函數(shù))

Python中如何實(shí)現(xiàn)運(yùn)算符的重載,即實(shí)現(xiàn)例如a+b這樣的運(yùn)算符操作呢?

在C++中可以使用 operator 關(guān)鍵字實(shí)現(xiàn)運(yùn)算符的重載。但是在Python中沒(méi)有類(lèi)似這樣的關(guān)鍵字,所以要實(shí)現(xiàn)運(yùn)算符的重載,就要用到Python的魔法函數(shù)。Python魔法函數(shù)是以雙下劃線(xiàn)開(kāi)頭,雙下劃線(xiàn)結(jié)尾的一組函數(shù)。我們?cè)陬?lèi)定義中最常用到的 __init__ 函數(shù)就是這樣一個(gè)魔法函數(shù),它在創(chuàng)建類(lèi)對(duì)象時(shí)被自動(dòng)調(diào)用。

下面我們來(lái)看個(gè)簡(jiǎn)單的例子。

上述代碼示例了幾個(gè)魔法函數(shù)的用法。 __add__ 函數(shù)對(duì)應(yīng)了二元運(yùn)算符+,當(dāng)執(zhí)行a+b語(yǔ)句時(shí),python就會(huì)自動(dòng)調(diào)用a. add (b)。 對(duì)于上述例子中的v1+v2+v3,則相當(dāng)于調(diào)用了(v1. add(v2)). add(v3)。

代碼中還有一個(gè)在Python類(lèi)定義經(jīng)常使用的 __str__ 函數(shù),當(dāng)使用 str() 時(shí)會(huì)被調(diào)用。print函數(shù)對(duì)傳入的參數(shù)都調(diào)用了str()將其轉(zhuǎn)換成易讀的字符串形式,便于打印輸出,因而會(huì)調(diào)用類(lèi)定義的__str__函數(shù)打出自定義的字符串。

代碼中還有一個(gè)特殊的 __call__ 函數(shù),該函數(shù)在將對(duì)象采用函數(shù)調(diào)用方式使用時(shí)被調(diào)用, 例如v1()相當(dāng)于v1. call ()。

以上就是魔法函數(shù)的基本使用方法。常見(jiàn)的魔法函數(shù)我們可以使用 dir() 函數(shù)來(lái)查看。

輸出結(jié)果為:

上述結(jié)果中形式為‘__函數(shù)名__’的函數(shù)為魔法函數(shù),注意有些對(duì)象也是這種形式,例如__class__, __module__等, 這些不是魔法函數(shù)。具體的魔法函數(shù)說(shuō)明可以參考Python官方說(shuō)明文檔。

以上代碼在Python3.6運(yùn)行通過(guò).

【python】魔法方法 :__getitem__ 、 __len__、__setitem__等的使用

在Python中,如果我們想實(shí)現(xiàn)創(chuàng)建類(lèi)似于序列和映射的類(lèi)(可以迭代以及通過(guò)[下標(biāo)]返回元素),可以通過(guò)重寫(xiě)魔法方法 __getitem__、__setitem__、__delitem__、__len__ 方法去模擬。

__getitem__(self,key): 返回鍵對(duì)應(yīng)的值。

__setitem__(self,key,value): 設(shè)置給定鍵的值

__delitem__(self,key): 刪除給定鍵對(duì)應(yīng)的元素。

__len__(): 返回元素的數(shù)量

【注釋】只要實(shí)現(xiàn)了 __getitem__ 和 __len__ 方法,就會(huì)被認(rèn)為是序列。

這些魔術(shù)方法的原理就是:當(dāng)我們對(duì)類(lèi)的屬性item進(jìn)行下標(biāo)的操作時(shí),首先會(huì)被 __getitem__()、__setitem__()、__delitem__() 攔截,從而執(zhí)行我們?cè)诜椒ㄖ性O(shè)定的操作,如賦值,修改內(nèi)容,刪除內(nèi)容等等。

這個(gè)方法應(yīng)該以與鍵相關(guān)聯(lián)的方式存儲(chǔ)值,以便之后能夠使用 __setitem__ 來(lái)獲取。當(dāng)然,這個(gè)對(duì)象可變時(shí)才需要實(shí)現(xiàn)這個(gè)方法。

舉個(gè)栗子:

定義一副撲克牌(不包括大小王),對(duì)牌進(jìn)行洗牌,然后發(fā)牌。

Output:

【注意】 :我們會(huì)發(fā)現(xiàn)output中,輸出了: slice(1, 3, None) ,下面給出解釋。

語(yǔ)法:

參數(shù)說(shuō)明:

slice() 函數(shù)實(shí)現(xiàn)切片對(duì)象,主要用在切片操作函數(shù)里的參數(shù)傳遞。

舉兩個(gè)栗子來(lái)看看:

Output:

切片原理

output

(程序員必會(huì)的 hhhhh.....)

看看slice在python3.7中是怎么描述的:

當(dāng)前文章:魔法函數(shù)python 魔法函數(shù)Python為啥報(bào)錯(cuò)
網(wǎng)站地址:http://muchs.cn/article18/hgsjdp.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供全網(wǎng)營(yíng)銷(xiāo)推廣微信小程序、手機(jī)網(wǎng)站建設(shè)、動(dòng)態(tài)網(wǎng)站、企業(yè)網(wǎng)站制作營(yíng)銷(xiāo)型網(wǎng)站建設(shè)

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶(hù)投稿、用戶(hù)轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀(guān)點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話(huà):028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)

網(wǎng)站優(yōu)化排名