九游app:PHP中的常見魔術(shù)方法功能作用及用法實(shí)例
在面向?qū)ο缶幊讨?,PHP提供了一系列的魔術(shù)方法,這些魔術(shù)方法為編程提供了很多便利。PHP中的魔術(shù)方法通常以__(兩個(gè)下劃線)開始,并且不需要顯示的調(diào)用而是由某種特定的條件出發(fā)。這篇文章簡單總結(jié)了PHP中提供的魔術(shù)方法。
Device類有四個(gè)成員屬性和兩個(gè)成員方法。Battery類有一個(gè)成員屬性和一個(gè)成員方法。
構(gòu)造函數(shù)和析構(gòu)函數(shù)分別在對象創(chuàng)建和銷毀時(shí)被調(diào)用。對象被“銷毀”是指不存在任何對該對象的引用,比如引用該對象的變量被刪除(unset)、重新賦值或腳本執(zhí)行結(jié)束,都會(huì)調(diào)用析構(gòu)函數(shù)。
__construct()構(gòu)造函數(shù)是目前為止最經(jīng)常使用的函數(shù)。在創(chuàng)建對象時(shí),可以在構(gòu)造函數(shù)中做一些初始化工作??梢詾闃?gòu)造函數(shù)定義任意多個(gè)參數(shù),只要在實(shí)例化時(shí)傳入對應(yīng)個(gè)數(shù)的參數(shù)即可。構(gòu)造函數(shù)中出現(xiàn)的任何異常都會(huì)阻止對象的創(chuàng)建。
上面的示例代碼中,Device類的構(gòu)造函數(shù)為成員屬性賦值并且調(diào)用了connect()方法。
將構(gòu)造函數(shù)聲明為私有方法,可以防止在類外部創(chuàng)建對象,這在單利模式中經(jīng)常使用。
析構(gòu)函數(shù)通常在對象被銷毀時(shí)調(diào)用,析構(gòu)函數(shù)不接收任何參數(shù)。經(jīng)常在析構(gòu)函數(shù)中執(zhí)行一些清理工作,比如關(guān)閉數(shù)據(jù)庫連接等。
有一點(diǎn)需要注意的是:PHP中的”重載”與其他大多數(shù)語言的重載不是太一樣,雖然都實(shí)現(xiàn)了相同的功能。
屬性重載涉及到的兩個(gè)魔術(shù)方法主要是用來處理屬性訪問,定義了當(dāng)我們嘗試訪問一個(gè)不存在(或不可訪問)的屬性時(shí)會(huì)發(fā)生什么。
魔術(shù)方法__get()在我們嘗試訪問一個(gè)不存在的屬性時(shí)會(huì)被調(diào)用。它接收一個(gè)參數(shù),該參數(shù)表示訪問屬性的名字,并且將該屬性的值返回。在上面的Device類里,有一個(gè)data屬性,該屬性就在這里就起了作用,如下面得代碼:
該魔術(shù)方法最常用的地方就是通過創(chuàng)建一個(gè)“只讀”的屬性來擴(kuò)展訪問控制。在上面的Battery類中,有一個(gè)私有屬性$charge,我們可以通過__get()魔術(shù)方法將該屬性擴(kuò)展為在類外部可讀但不能修改。代碼如下:
__set()魔術(shù)方法在我們嘗試修改一個(gè)不可訪問的屬性時(shí)會(huì)被調(diào)用,它接收兩個(gè)參數(shù),一個(gè)表示屬性的名字,一個(gè)表示屬性的值。示例代碼如下:
__isset()魔術(shù)方法在對一個(gè)不可訪問的屬性調(diào)用isset()方法時(shí)會(huì)被調(diào)用,它接收一個(gè)參數(shù),表示屬性的名字。它應(yīng)該返回一個(gè)布爾值,用來表示該屬性是否存在。代碼如下:
__unset()魔術(shù)方法在調(diào)用unset()函數(shù)銷毀一個(gè)不能訪問的屬性時(shí)會(huì)被調(diào)用,它接收一個(gè)參數(shù),表述屬性的名字。
__toString()在我們將對象當(dāng)作字符串一樣使用時(shí)會(huì)被調(diào)用,它不接收任何參數(shù)。該方法允許我們定義對象的表現(xiàn)形式。代碼如下:
靜態(tài)魔術(shù)方法__set_state(),在我們使用var_export()函數(shù)輸出對象時(shí)會(huì)調(diào)用該方法。var_export()函數(shù)用來將PHP變量轉(zhuǎn)換為PHP代碼,它接收一個(gè)包含對象屬性值的關(guān)聯(lián)數(shù)組作為參數(shù)。示例代碼如下:
默認(rèn)的,對象都是按引用傳值的。因此,在將一個(gè)對象賦值給另一個(gè)變量時(shí),只是創(chuàng)建了指向該對象的一個(gè)引用,并沒有復(fù)制該對象。為了實(shí)現(xiàn)真正得復(fù)制一個(gè)對象,我們需要使用clone關(guān)鍵字。
這種“按引用傳遞”的策略同樣適用于包含在對象內(nèi)部的對象。即使我們克隆了一個(gè)對象,在對象內(nèi)部的任何對象都不會(huì)被克隆,因此最終的結(jié)果是兩個(gè)對象共享了同一個(gè)內(nèi)部對象。示例代碼如下:
__clone()魔術(shù)方法__clone()可以解決上面的問題。當(dāng)對一個(gè)對象使用clone關(guān)鍵字時(shí),該魔術(shù)方法會(huì)被調(diào)用。在這個(gè)魔術(shù)方法里,我們可以實(shí)現(xiàn)任何子對象的克隆,代碼如下:
序列化是講任意數(shù)據(jù)轉(zhuǎn)換為字符串格式的過程。序列化通常用來將整個(gè)對象存入數(shù)據(jù)庫或?qū)懭胛募?。?dāng)反序列化存儲(chǔ)的數(shù)據(jù)時(shí),我們可以得到序列化之前的對象。但是,并不是所有得數(shù)據(jù)都可以被序列化,比如數(shù)據(jù)庫連接。幸運(yùn)的是,有一個(gè)魔術(shù)方法可以幫我們解決這個(gè)問題。
魔術(shù)方法__sleep()在對一個(gè)對象序列化時(shí)(調(diào)用serialize())會(huì)被調(diào)用。它不接收任何參數(shù),而且應(yīng)該返回一個(gè)包含所有應(yīng)該被序列化的屬性的數(shù)組。在該魔術(shù)方法中,也可以執(zhí)行一些其他操作。
有一點(diǎn)要注意的是,不要再該函數(shù)中進(jìn)行任何的析構(gòu)操作,因?yàn)檫@可能會(huì)影響正在運(yùn)行的對象。
魔術(shù)方法__wakeup()在對存儲(chǔ)的對象反序列化時(shí)會(huì)被調(diào)用。它不接收任何參數(shù),也沒有任何返回值??梢杂盟鼇硖幚碓谛蛄谢瘯r(shí)丟失的數(shù)據(jù)庫連接或資源。代碼如下:
PHP還有兩個(gè)與成員方法相關(guān)的魔術(shù)方法__call()和__callStatic(),這兩個(gè)魔術(shù)方法類似于屬性重載方法。
魔術(shù)方法__call()在調(diào)用不存在或不可訪問的方法時(shí)會(huì)被調(diào)用。它接收兩個(gè)參數(shù),一個(gè)是調(diào)用的方法的名字,一個(gè)是包含函數(shù)參數(shù)的數(shù)組。我們可以使用這種方法調(diào)用子對象中得同名函數(shù)。
在這個(gè)例子中,要注意函數(shù)call_user_func_array(),這個(gè)函數(shù)允許我們動(dòng)態(tài)調(diào)用一個(gè)命名的函數(shù)。
魔術(shù)方法__callStatic()與__call()的功能一樣,唯一不同的是,該方法在嘗試訪問一個(gè)不存在或不可訪問的靜態(tài)方法時(shí)會(huì)被調(diào)用。示例代碼如下:
有時(shí)候我們會(huì)需要將對象作為函數(shù)使用。將對象作為函數(shù)使用,就像我們使用其他普通的函數(shù)一樣,可以傳參。
魔術(shù)方法__invoke()在嘗試將對象作為函數(shù)使用時(shí)會(huì)被調(diào)用。在該方法中定義的任何參數(shù),都將被作為函數(shù)的參數(shù)。示例代碼如下:
__autoload()方法并不是一個(gè)魔術(shù)方法,但是這個(gè)方法非常有用。但是,對著PHP版本的更新,該函數(shù)已經(jīng)不建議使用,取而代之的是spl_auto_register()函數(shù)。九游娛樂-平臺(tái)官網(wǎng)