PHP获取类私有属性的3种方法
作者:tlanyan 发布时间:2023-11-20 10:32:16
今天在推上看到一条获取PHP类私有属性的推文,感觉很有意思:
顺着推文联想,还有其他方式吗?经过自己的测试及网上答案,总结出三种方法:
1. 反射
反射可以获取类的详细信息,要获取私有属性的值,只需将对应属性的ReflectionProperty实例设置为可访问再取值即可。示例代码如下:
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. 转换成数组
这种方法用将对象强制转换成数组,再通过键获取其值。示例代码如下:
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
让其现形:
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
的用法示例如下:
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
方法的第二个参数注意传入对象的 “完全限定类名”,指示函数应该放置在该类的作用域下,从而可以访问私有属性。
总结
性能: 数组 > 反射 > 闭包
易用性: 闭包 > 数组 > 反射
推荐: 闭包 > 反射 > 数组
来源:https://tlanyan.me/ways-to-access-php-class-private-members/
猜你喜欢
- 问题在一个python web应用中需要定时执行一些任务,所以用了APScheduler这个库。又因为是用flask这个web框架,所以用了
- Python 操作文件时,我们一般要先判断指定的文件或目录是否存在,不然容易产生异常。1.文件# 是否存在import osos.path.
- 之前用python调用API存JSON的时候试用了很多方法,现在调用API直接获取参数的时候也是查了好多例子(毕竟我是一个初学者)。结果让我
- css实现的圆角矩形的方式很多,但要追求灵活型,上面的结构简单,看起来爽一点注意css所用的图片路径,已修改兼容ie6 ie7 ff ,IE
- eclare @str nvarchar(50);set @str='462,464,2';select @str as &
- 写在前面今天在公司写了一段代码,判断一个变量是否为空值,由于判断的类型太少,code review的时候同事说还有很多类型没有考虑到,并且提
- 一、结论语法结构: limit offset, rows结论:rows 相同条件下,offset 值越大,limit 语句性能越差二、测试执
- sql不常用函数总结以及事务,增加,删除触发器 distinct 删除重复行 declare @x 申明一个变量 convert(varch
- 1. 很多情况下需要进行不同进制之间的转换其中比较常用到的是python的内置函数进行进制的转换,一般使用内置函数进行转换的时候是先将控制台
- 看看下面的w3 upload组件例子,很简单: upload.asp<html> <head>
- 目录1. 递归函数与回溯深搜的基础知识2. 求子集 (LeetCode 78)3. 求子集2 (LeetCode 90)4. 组合数之和(L
- python文件I/O打印到屏幕:最简单的输出方法是用print语句,你可以给它传递零个或多个用逗号隔开的表达式。读取键盘输入:Python
- 需求每天请求一封邮件,并读取该邮件这个其实可以使用linux 自带了crontab实现,但是毕竟是django 开发。想着不知道有没有方法可
- 将ubk_vhost_list表中的字段userid中的字符10005替换成10010 UPDATE `table_name` SET `f
- HTML代码如下,其中,要拖动的div为最外层的div <div id="dialog_createUserGroup&qu
- Jupyter Notebooks 是什么?Jupyter Notebooks 是一款开源的网络应用,我们可以将其用于创建和共享代码与文档。
- 本文实例对比分析了python中lambda与def的用法。分享给大家供大家参考。具体分析如下:1、lambda用来创建匿名函数,不同于de
- 无法远程登入MySQL数据库的几种解决办法方法一:尝试用MySQL Adminstrator GUI Tool登入MySQL Server,
- 直接调用系统的颜色显示在网页上本来是件很好玩滴事,但是,也有个缺点,就是可用的色太少 比如Bindows在它的启动画面一点点应用。=。= 上
- 我们先来看一下效果(简单的写了一个):原理:将post请求的代码数据写入了服务器的一个文件,然后用服务器的python编译器执行返回结果实现