PHP獲取類私有屬性的幾種方式

轉載請註明文章出處:https://tlanyan.me/ways-to-ac...

今天在推上看到一條獲取PHP類私有屬性的推文,感覺很有意思:

img

順着推文聯想,還有其他方式嗎?經過自己的測試及網上答案,總結出三種方法:

1. 反射

反射可以獲取類的詳細信息,要獲取私有屬性的值,只需將對應屬性的ReflectionProperty實例設置爲可訪問再取值即可。示例代碼如下:

<pre>namespace tlanyan;

class Foo {
private $bar = "Foo bar!";
}

// 獲取反射類及反射屬性
$class = new ReflectionClass(Foo::class);
$property = $class->getProperty("bar");
// 設置屬性可訪問
$property->setAccessible(true);

$foo = new Foo;
// 獲取對象屬性值
// 注意:只能通過 ReflectionProperty 實例的 getValue 方法訪問
// 不能這樣直接訪問: $foo->bar;
echo $property-&gt;getValue($foo), PHP_EOL:
// 輸出: Foo bar!
</pre>

本人之前寫過“PHP回顧之反射”一文,比較詳細的介紹了反射及用法,有興趣的閱讀參考。

2. 轉換成數組

這種方法用將對象強制轉換成數組,再通過鍵獲取其值。示例代碼如下:

<pre>class Foo {
private $bar = "Foo bar!";
}

$foo = new Foo;
// 強制轉型
$attrs = (array)$foo;
// 拼接key,注意 "0" 不能改成單引號!
$key = "0" . Foo::class . "0" . "bar";
echo $attrs[$key], PHP_EOL;
// 輸出: Foo bar!
</pre>

上述代碼中key的拼接方式比較詭異,根據鳥哥“深入理解PHP原理之對象(一)”文中的信息,key規則如下:

  1. public屬性, key是 屬性名
  2. protected屬性,key是 \0*\0屬性名
  3. private屬性, key是 \0類名\0屬性名

注意 \0 是一個字符(不是兩個),對應的ASCII碼是數字0。編程時要用雙引號將其引起來。不能使用單引號,否則轉義失效,那就是兩個字符。如果你有C語言基礎,應該知道 \0 就是字符串的結束符。這個符號直接輸出不會顯示,但可以通過strlen或者ord讓其現形:

<pre>foreach ($attrs as $key => $value) {
echo "key:$key", ", key length:", strlen($key), ", ascii: ";
for ($i = 0; $i < strlen($key); ++ $i) {

echo ord($key[$i]), " "; 

}
echo PHP_EOL;
}
// 輸出
// key:Foobar, key length:8, ascii: 0 70 111 111 0 98 97 114
// Foobar 有6個字符,加上兩個不顯示字符,所以長度是8
</pre>

還需要注意拼接private屬性時類名應該是 “完全限定類名” ,建議通過Foo::class的方式獲取。

與強制轉換成數組類似的另一種方法是serialize,但是serialize比較慢,並且序列化後的字符串更難辨認結構和處理,不建議使用。

3. 閉包

文章開頭的推特截圖已經展示了閉包的用法,其中call方法在PHP7中引入,另一個是PHP5.4引入的bindTocallbindTo的用法示例如下:

<pre>namespace tlanyan;

class Foo {
private $bar = "Foo bar!";
}

$foo = new Foo;
// 閉包(匿名函數)是PHP5.3引入的功能
$closure = function() { return $this->bar; };
// PHP5.4起支持bindTo方法
$method = $closure->bindTo($foo, Foo::class);
echo $method(), PHP_EOL;

// PHP7引入call方法,可綁定this直接執行
echo $closure-&gt;call($foo), PHP_EOL;
</pre>

bindTo方法的第二個參數注意傳入對象的 “完全限定類名”,指示函數應該放置在該類的作用域下,從而可以訪問私有屬性。

總結

性能: 數組 > 反射 > 閉包

易用性: 閉包 > 數組 > 反射

推薦: 閉包 > 反射 > 數組

參考

  1. 深入理解PHP原理之對象(一)
  2. Accessing private properties in PHP
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章