PHP获取类私有属性的几种方式

时间: 2019-07-10阅读: 370标签: 属性

今天在推上看到一条获取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的拼接方式比较诡异,根据鸟哥“ 深入理解PHP原理之对象(一) ”文中的信息,key规则如下:

  1. public 属性, key是 属性名 ;
  2. protected 属性,key是 \0*\0属性名 ;
  3. 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://www.tlanyan.me/ways-to-access-php-class-private-members/


吐血推荐

1.站长广告联盟: 整理了目前主流的广告联盟平台,如果你有流量,可以作为参考选择适合你的平台点击进入...

2.休闲娱乐: 网页游戏  直播/交友   H5游戏

链接: http://www.fly63.com/article/detial/4616

Vue.js 计算属性

在html模版中绑定数据时,如果需要的数据不是一个简单的数据,而是需要将一个或者多个数据计算之后得到的数据,这时候就需要用到计算属性;使用方法:配置computed属性,它的值为对象

vue中计算属性的get与set方法

在vue的计算属性中,所定义的都是属性,可以直接调用,正常情况下,计算属性中的每一个属性对应的都是一个对象,对象中包括了set方法与get方法

CSS3的all属性

all属性实际上是所有CSS属性的缩写,表示,所有的CSS属性都怎样怎样,但是,不包括unicode-bidi和direction这两个CSS属性。为什么会有这个CSS属性呢?

CSS Display属性的双值写法

目前这个只有Firefox70支持了这一语法,其他的浏览器仍将其当成非法的语法处理,因此生产情况下使用还是为时过早。目前所有的功能都可以用单一值来实现,所以这个新的语法可能会作为别名的形式存在

Js中的offsetWidth、clientWidth、innerWidth及相关属性方法

下面的元素属性和元素方法都通过 elem.属性 或 elem.方法 的方式使用,window属性通过 window.属性 的方式使用,document属性则通过document调用:

HTML 标签的 for 属性

for 属性规定 label 与哪个表单元素绑定。隐式和显式的联系label通常以下面两种方式中的一种来和表单控件相联系:将表单控件作为标记标签的内容,这样的就是隐式形式

CSS属性white-space

本文说的white-space是一个控制换行和空白处理的CSS属性。我曾经被这个属性烦死,一直没记住,今天决定还是写下来好好琢磨下。normal默认值,正常换行,空白和换行符会被浏览器忽略。啥意思呢?

html5全局属性

accesskey设置访问元素的键盘快捷键。class规定元素的类名(classname),contenteditable规定是否可编辑元素的内容。contextmenu 指定一个元素的上下文菜单

React 也能“用上” computed属性

初次见到计算属性一词,是在 Vue 官方文档 《计算属性和侦听器》 一节中,文章中是这样描述计算属性的:模板内的表达式非常便利,但是设计它们的初衷是用于简单运算的。在模板中放入太多的逻辑会让模板过重且难以维护。

CSS box-sizing属性的正确用法

CSS box-sizing属性允许我们在元素的总宽度和高度中包含填充和边框。默认情况下,元素的宽度和高度计算如下: width + padding + border =元素的实际宽度 height + padding + border =元素的实际高度 这意味着

fly63.com版权所有,内容以共享、参考、研究为目的,不存在任何商业目的。其版权属原作者所有,如有侵权,请与小编联系!情况属实本人将予以删除!

广告赞助文章投稿关于web前端网站点搜索站长推荐网站地图站长QQ:522607023

小程序专栏: 土味情话心理测试脑筋急转弯幽默笑话段子句子语录成语大全