PHP获取类私有属性的3种方法

这篇文章主要介绍了PHP获取类私有属性的3种方法,帮助大家更好的理解和使用php,感兴趣的朋友可以了解下。

今天在推上看到一条获取PHP类私有属性的推文,感觉很有意思:

顺着推文联想,还有其他方式吗?经过自己的测试及网上答案,总结出三种方法:

1. 反射

反射可以获取类的详细信息,要获取私有属性的值,只需将对应属性的ReflectionProperty实例设置为可访问再取值即可,示例代码如下:

  1. namespace tlanyan;
  2. class Foo {
  3. private $bar = "Foo bar!";
  4. }
  5. // 获取反射类及反射属性
  6. $class = new \ReflectionClass(Foo::class);
  7. $property = $class->getProperty("bar");
  8. // 设置属性可访问
  9. $property->setAccessible(true);
  10. $foo = new Foo;
  11. // 获取对象属性值
  12. // 注意:只能通过 ReflectionProperty 实例的 getValue 方法访问
  13. // 不能这样直接访问: $foo->bar;
  14. echo $property->getValue($foo), PHP_EOL:
  15. // 输出: Foo bar!

本人之前写过“PHP回顾之反射”一文,比较详细的介绍了反射及用法,有兴趣的阅读参考。

2. 转换成数组

这种方法用将对象强制转换成数组,再通过键获取其值,示例代码如下:

  1. class Foo {
  2. private $bar = "Foo bar!";
  3. }
  4. $foo = new Foo;
  5. // 强制转型
  6. $attrs = (array)$foo;
  7. // 拼接key,注意 "\0" 不能改成单引号!
  8. $key = "\0" . Foo::class . "\0" . "bar";
  9. echo $attrs[$key], PHP_EOL;
  10. // 输出: Foo bar!

上述代码中key的拼接方式比较诡异,key规则如下:

public属性, key是 属性名;

protected属性,key是 \0*\0属性名;

private属性, key是 \0类名\0属性名。

注意 \0 是一个字符(不是两个),对应的ASCII码是数字0,编程时要用双引号将其引起来。不能使用单引号,否则转义失效,那就是两个字符,如果你有C语言基础,应该知道 \0 就是字符串的结束符,这个符号直接输出不会显示,但可以通过strlen或者ord让其现形:

  1. foreach ($attrs as $key => $value) {
  2. echo "key:$key", ", key length:", strlen($key), ", ascii: ";
  3. for ($i = 0; $i < strlen($key); ++ $i) {
  4. echo ord($key[$i]), " ";
  5. }
  6. echo PHP_EOL;
  7. }
  8. // 输出
  9. // key:Foobar, key length:8, ascii: 0 70 111 111 0 98 97 114
  10. // Foobar 有6个字符,加上两个不显示字符,所以长度是8

还需要注意拼接private属性时类名应该是 “完全限定类名” ,建议通过Foo::class的方式获取。

与强制转换成数组类似的另一种方法是serialize,但是serialize比较慢,并且序列化后的字符串更难辨认结构和处理,不建议使用。

3. 闭包

文章开头的推特截图已经展示了闭包的用法,其中call方法在PHP7中引入,另一个是PHP5.4引入的bindTo。call和bindTo的用法示例如下:

  1. namespace tlanyan;
  2. class Foo {
  3. private $bar = "Foo bar!";
  4. }
  5. $foo = new Foo;
  6. // 闭包(匿名函数)是PHP5.3引入的功能
  7. $closure = function() { return $this->bar; };
  8. // PHP5.4起支持bindTo方法
  9. $method = $closure->bindTo($foo, Foo::class);
  10. echo $method(), PHP_EOL;
  11. // PHP7引入call方法,可绑定this直接执行
  12. echo $closure->call($foo), PHP_EOL;

bindTo方法的第二个参数注意传入对象的 “完全限定类名”,指示函数应该放置在该类的作用域下,从而可以访问私有属性。

总结

性能: 数组 > 反射 > 闭包

易用性: 闭包 > 数组 > 反射

推荐: 闭包 > 反射 > 数组