今天在推上看到一条获取PHP类私有属性的推文,感觉很有意思:
顺着推文联想,还有其他方式吗?经过自己的测试及网上答案,总结出三种方法:
1. 反射
反射可以获取类的详细信息,要获取私有属性的值,只需将对应属性的ReflectionProperty实例设置为可访问再取值即可。示例代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
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 ->getValue( $foo ), PHP_EOL:
// 输出: Foo bar!
|
本人之前写过“PHP回顾之反射”一文,比较详细的介绍了反射及用法,有兴趣的阅读参考。
2. 转换成数组
这种方法用将对象强制转换成数组,再通过键获取其值。示例代码如下:
1
2
3
4
5
6
7
8
9
10
11
|
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!
|
上述代码中key的拼接方式比较诡异,key规则如下:
-
public
属性, key是 属性名; -
protected
属性,key是 \0*\0属性名; -
private
属性, key是 \0类名\0属性名。
注意 \0 是一个字符(不是两个),对应的ASCII码是数字0。编程时要用双引号将其引起来。不能使用单引号,否则转义失效,那就是两个字符。如果你有C语言基础,应该知道 \0 就是字符串的结束符。这个符号直接输出不会显示,但可以通过strlen
或者ord
让其现形:
1
2
3
4
5
6
7
8
9
10
|
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
|
还需要注意拼接private
属性时类名应该是 “完全限定类名” ,建议通过Foo::class
的方式获取。
与强制转换成数组类似的另一种方法是serialize
,但是serialize
比较慢,并且序列化后的字符串更难辨认结构和处理,不建议使用。
3. 闭包
文章开头的推特截图已经展示了闭包的用法,其中call
方法在PHP7中引入,另一个是PHP5.4引入的bindTo
。call
和bindTo
的用法示例如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
|
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 ->call( $foo ), PHP_EOL;
|
bindTo
方法的第二个参数注意传入对象的 “完全限定类名”,指示函数应该放置在该类的作用域下,从而可以访问私有属性。
总结
性能: 数组 > 反射 > 闭包
易用性: 闭包 > 数组 > 反射
推荐: 闭包 > 反射 > 数组
以上就是PHP获取类私有属性的3种方法的详细内容,更多关于PHP获取类私有属性的资料请关注服务器之家其它相关文章!
原文链接:https://tlanyan.me/ways-to-access-php-class-private-members/