PHP自從5.3版以來就新增了一個叫做__invoke的魔術方法,使用該方法就可以在創建實例後,直接調用對象。如下示例所示:
class testClass { public function __invoke { print "hello world"; } } $n = new testClass; $n();
執行結果為:
hello world。
php官方示例如下:
class CallableClass { public function __invoke($x) { var_dump($x); } } $obj = new CallableClass; $obj(5); var_dump(is_callable($obj));
附:PHP其他魔術方法詳解:
php中的類就可以使用魔術方法了。其規定以兩個下劃線(__)開頭的方法都保留為魔術方法,所以建議大家函數名最好不用__開頭,除非是為了重載已有的魔術方法。
The function names __construct, __destruct, __call, __callStatic, __get, __set, __isset, __unset, __sleep, __wakeup, __toString, __invoke, __set_state and__clone are magical in PHP classes. You cannot have functions with these names in any of your classes unless you want the magic functionality associated with them.
如下這兩個方法是為在類和他們的父類中沒有聲明的屬性而設計的。
__get($property)當調用一個未定義的屬性時,此方法會被觸發,傳遞的參數是被訪問的屬性名。
__set($property,$value)給一個未定義的屬性賦值時,此方法會被觸發,傳遞的參數是被設置的屬性名和值。
這裡的沒有聲明包括當使用對象調用時,訪問控制為proteced,private的屬性(即沒有權限訪問的屬性)。
與__get方法和__set方法相同,這裡的沒有聲明包括當使用對象調用時,訪問控制為proteced,private的屬性(即沒有權限訪問的屬性)。
這裡的未定義的方法包括沒有權限訪問的方法;如果方法不存在就去父類中找這個方法,如果父類中也不存在就去調用本類的__call()方法,如果本類中不存在__call()方法就去找父類中的__call()方法。
__autoload函數,它會在試圖使用尚未被定義的類時自動調用。通過調用此函數,腳本引擎在php出錯失敗前有了最後一個機會加載所需的類。
注意:在__autoload函數中拋出的異常不能被catch語句塊捕獲並導致致命錯誤,所以應該在函數本身做捕獲。
__construct構造方法,當一個對象創建時調用此方法,相對於php4使用此方法的好處是:可以使構造方法有一個獨一無二的名稱,無論它所在的類的名稱是什麼.這樣你在改變類的名稱時,就不需要改變構造方法的名稱。
__destruct析構方法,php將在對象被銷毀前(即從內存中清除前)調用這個方法。默認情況下,php僅僅釋放對象屬性所占用的內存並銷毀對象相關的資源,析構函數允許你在使用一個對象之後執行任意代碼來清除內存。當php決定你的腳本不再與對象相關時,析構函數將被調用。
在一個函數的命名空間內,這會發生在函數return的時候。對於全局變量,這發生於腳本結束的時候。如果你想明確地銷毀一個對象,你可以給指向該對象的變量分配任何其它值.通常將變量賦值為null或者調用unset。
php5中的對象賦值是使用的引用賦值,如果想復制一個對象則需要使用clone方法,在調用此方法是對象會自動調用__clone魔術方法,如果在對象復制需要執行某些初始化操作,可以在__clone方法實現。
__tostring方法在將一個對象轉化成字符串時自動調用,比如使用echo打印對象時。
在php5.2.0之前,__tostring方法只有結合使用echo()或print()時才能生效。php5.2.0之後,則可以在任何字符串環境生效(例如通過printf(),使用%s修飾符),但不能用於非字符串環境(如使用%d修飾符)。從php5.2.0,如果將一個未定義__tostring方法的對象轉換為字符串,會報出一個e_recoverable_error錯誤。
__wakeup反串行化的時候調用
serialize()檢查類中是否有魔術名稱__sleep的函數。如果這樣,該函數將在任何序列化之前運行。它可以清除對象並應該返回一個包含有該對象中應被序列化的所有變量名的數組。
使用__sleep的目的是關閉對象可能具有的任何數據庫連接,提交等待中的數據或進行類似的清除任務。此外,如果有非常大的對象而並不需要完全儲存下來時此函數也很有用。
PHP相反地,unserialize()檢查具有魔術名稱__wakeup的函數的存在。如果存在,此函數可以重建對象可能具有的任何資源。使用__wakeup的目的是重建在序列化中可能丟失的任何數據庫連接以及處理其它重新初始化的任務。
當嘗試以調用函數的方式調用一個對象時,__invoke方法會被自動調用。
它的工作方式類似於__call()魔術方法,__callstatic()是為了處理靜態方法調用。
php確實加強了對__callstatic()方法的定義;它必須是公共的,並且必須被聲明為靜態的。同樣,__call()魔術方法必須被定義為公共的,所有其他魔術方法都必須如此。
這樣繞的話是為了安全,也就是面向對象裡封裝的理念,如果直接設置Public那麼沒個對象都可以自由設置這個屬性的值了,而且沒有經過任務邏輯判斷,這樣繞彎的話可以增加安全性
PHP處理對象部分的內核完全重新開發過,提供更多功能的同時也提高了性能。在以前版本的php中,處理對象和處理基本類型(數字,字符串)的方式是一樣的。這種方式的缺陷是:當將對象賦值給一個變量時,或者通過參數傳遞對象時,對象將被完全拷貝一份。在新的版本裡,上述操作將傳遞引用(可以把引用理解成對象的標識符),而非值。
很多PHP程序員可能甚至沒有察覺到老的對象處理方式。事實上,大多數的php應用都可以很好地運行。或者僅僅需要很少的改動。
私有和受保護成員
PHP5引入了私有和受保護成員變量的概念。我們可以用它來定義類成員的可見性。
例子
受保護成員可以被子類訪問, 而私有成員只能被類本身訪問。
代碼:--------------------------------------------------------------------------------
<?php
class MyClass {
private $Hello = "Hello, World!\n";
protected $Bar = "Hello, Foo!\n";
protected $Foo = "Hello, Bar!\n";
function printHello() {
print "MyClass::printHello() " . $this->Hello;
print "MyClass::printHello() " . $this->Bar;
print "MyClass::printHello() " . $this->Foo;
}
}
class MyClass2 extends MyClass {
protected $Foo;
function printHello() {
MyClass::printHello(); /* Should print */
print "MyClass2::printHello() " . $this->Hello; /* Shouldn't print out anything */
print "MyClass2::printHello() " . $this->Bar; /* Shouldn't print (not declared)*/
print "MyClass2::printHello() " . $this->Foo; /* Should print */
}
}
$obj = new MyClass();
print $obj->Hello; /* Shouldn't print out anything */
print $obj->Bar; /* Shouldn't print out anything */
print $obj->Foo; /* Should......余下全文>>