1. 程式人生 > 程式設計 >PHP的過載使用魔術方法程式碼例項詳解

PHP的過載使用魔術方法程式碼例項詳解

摘錄php官網對PHP過載的解釋:

PHP所提供的"過載"(overloading)是指動態地"建立"類屬性和方法。我們是通過魔術方法(magic methods)來實現的。
當呼叫當前環境下未定義或不可見的類屬性或方法時,過載方法會被呼叫。本節後面將使用"不可訪問屬性(inaccessible properties)"和"不可訪問方法(inaccessible methods)"來稱呼這些未定義或不可見的類屬性或方法。
所有的過載方法都必須被宣告為 public。

Note:
這些魔術方法的引數都不能通過引用傳遞。

Note:
PHP中的"過載"與其它絕大多數面嚮物件語言不同。傳統的"過載"是用於提供多個同名的類方法,但各方法的引數型別和個數不同。

屬性過載

public __set ( string $name,mixed $value ) : void
public __get ( string $name ) : mixed
public __isset ( string www.cppcns.com$name ) : bool
public __unset ( string $name ) : void

在給不可訪問屬性賦值時,__set() 會被呼叫。
讀取不可訪問屬性的值時,__get() 會被呼叫。
當對不可訪問屬性呼叫 isset() 或 empty() 時,__isset() 會被呼叫。
當對不可訪問屬性呼叫 unset() 時,__unset() 會被呼叫。

引數 $name 程式設計客棧是指要操作的變數名稱。__set() 方法的 $value 引數指定了 $name 變數的值。
屬性過載只能在物件中進行。在靜態方法中,這些魔術方法將不會被呼叫。所以這些方法都不能被 宣告為 static。

Note:
因為 PHP 處理賦值運算的方式,__set() 的返回值將被忽略。類似的,在下面這樣的鏈式賦值中,__get() 不會被呼叫:
$a = $obj->b = 8;

Note:
在除 isset() 外的其它語言結構中無法使用過載的屬性,這意味著當對一個過載的屬性使用 empty() 時,過載魔術方法將不會被呼叫。
為避開此限制,必須將過載屬性賦值到本地變數再使用 empty()。

Example #1 使用 __get(),__set(),__issetsFtTHf() 和 __unset() 進行屬性過載

class PropertyTest
{
 /** 被過載的資料儲存在此 */
 private $data = array();

 /** 過載不能被用在已經定義的屬性 */
 public $declared = 1;

 /** 只有從類外部訪問這個屬性時,過載才會發生 */
 private $hidden = 2;

 public function __set ($name,$value)
 {
 $this->data[$name] = $value;
 }

 public function __get ($name)
 {
 if (isset($this->$name)) {
www.cppcns.com return $this->$name;
 }
 if (array_key_exists($name,$this->data)) {
 return $this->data[$name];
 }
 //產生一條回溯跟蹤
 $trace = debug_backtrace();
 //丟擲異常
 trigger_error('Undefined property via __get(): ' . $name . ' in ' . $trace[0]['file'] . ' on line ' . $trace[0]['line'],E_USER_NOTICE);
 return null;
 }

 public function __isset ($name)
 {
 return isset($this->data[$name]);
 }

 public function __unset ($name)
 {
 unset($this->data[$name]);
 }

 /** 非魔術方法 */
 public function getHidden ()
 {
 return $this->hidden;
 }
}

$obj = new PropertyTest;

//輸出不存在的a變數,走到__get()中,會丟擲異常
echo $obj->a;

//對不存在的a變數賦值為1,會走到__set()中
$obj->a = 1;

//再次輸出a變數,由於上面已經對其__set(),所以這是可以訪問到a的值為1
echo $obj->a;

//此時使用isset()對不存在的a變數進行運算時,會走到__isset()中,由於上面已經對其__set(),所以是true
var_dump(isset($obj->a));

//對a進行unset()時,會走到__unset()中
unset($obj->a);

//再對其進行isset(),此時已經不存在了
var_dump(isset($obj->a));

//訪問private 屬性的變數,會進入__get()中
echo $obj->hidden;

方法過載

public __call ( string $name,array $arguments ) : mixed
public static __callStatic ( string $name,array $arguments ) : mixed

在物件中呼叫一個不可訪問方法時,__call() 會被呼叫。
在靜態上下文中呼叫一個不可訪問方法時,__callStatic() 會被呼叫。
$name 引數是要呼叫的方法名稱。
$arguments 引數是一個列舉陣列,包含著要傳遞給方法 $name 的引數。

Example #2 使用 __call() 和 __callStatic() 對方法過載

class MethodTest
{
 /**
 * 呼叫不存在的方法時進入此處
 * sFtTHf@param $name
 * @param $arguments
 */
 public function __call ($name,$arguments)
 {
 // 注意: $name 的值區分大小寫
 $info = [
 'name' => $name,'arguments' => $arguments,];
 print_r($info);
 }

 /**
 * PHP 5.3.0之後版本
 * 呼叫不存在的靜態方法時,進入此處
 */
 public static function __callStatic ($name,];
 print_r($info);
 }
}

$arguments = ['A','B','C'];

$obj = new MethodTest;
$obj->test(...$arguments);

MethodTest::test(...$arguments); // PHP 5.3.0之後版本

/*
 * 以上兩個都輸出:
 * Array
 (
 [name] => test
 [arguments] => Array
 (
 [0] => A
 [1] => B
 [2] => C
 )

 )
 */

到此這篇關於PHP的過載使用魔術方法程式碼例項詳解的文章就介紹到這了,更多相關PHP的過載使用魔術方法內容請搜尋我們以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援我們!