有如下4个代码示例,你认为他们创建对象,并获得成员变量的速度排序是怎样的? 1:将成员变量设置为public,通过赋值操作给成员变量赋值,直接获取变量 class Foo public $id;$data = new Foo;$data-id = 10;echo $data-id; 2:将成员变量设置为public,通过
有如下4个代码示例,你认为他们创建对象,并获得成员变量的速度排序是怎样的?
1:将成员变量设置为public,通过赋值操作给成员变量赋值,直接获取变量
class Foo { public $id; } $data = new Foo; $data->id = 10; echo $data->id;
2:将成员变量设置为public,通过构造函数设置成员变量的值,直接获取变量
class Foo2 { public $id; public function __construct($id) { $this->id = $id; } } $data = new Foo2(10); echo $data->id;
3:将成员变量设置为protected,通过构造函数设置成员变量的值,通过成员方法获取变量
class Foo3 { protected $id; public function __construct($id) { $this->id = $id; } public function getId() { return $this->id; } } $data = new Foo3(10); echo $data->getId();
4:将成员变量设置为protected,通过构造函数设置成员变量的值,通过魔术方法获取变量
class Foo4 { protected $id; public function __construct($id) { $this->id = $id; } public function __get($key) { return $this->id; } } $data = new Foo4(10); echo $data->id;
按执行速度快慢排序: 1243
咱们先看其opcode:
1:
1 ZEND_FETCH_CLASS 4 :4 'Foo' 2 NEW $5 :4 3 DO_FCALL_BY_NAME 0 4 ASSIGN !0, $5 5 ZEND_ASSIGN_OBJ !0, 'id' 6 ZEND_OP_DATA 10 7 FETCH_OBJ_R $9 !0, 'id' 8 ECHO $9
2:
1 ZEND_FETCH_CLASS 4 :10 'Foo2' 2 NEW $11 :10 3 SEND_VAL 10 4 DO_FCALL_BY_NAME 1 5 ASSIGN !1, $11 6 FETCH_OBJ_R $14 !1, 'id' 7 ECHO $14
3:
1 ZEND_FETCH_CLASS 4 :15 'Foo3' 2 NEW $16 :15 3 SEND_VAL 10 4 DO_FCALL_BY_NAME 1 5 ASSIGN !2, $16 6 ZEND_INIT_METHOD_CALL !2, 'getId' 7 DO_FCALL_BY_NAME 0 $20 8 ECHO $20
4:
1 ZEND_FETCH_CLASS 4 :21 'Foo4' 2 NEW $22 :21 3 END_VAL 10 4 DO_FCALL_BY_NAME 1 5 ASSIGN !3, $22 6 FETCH_OBJ_R $25 !3, 'id' 7 ECHO $25
根据上面的opcode,参照其在zend_vm_execute.h文件对应的opcode实现,我们可以发现什么?
一、PHP内核创建对象的过程分为三步:
二、魔术方法的调用是通过条件触发的,并不是直接调用,如我们示例中的成员变量id的获取(zend_std_read_property),其步骤为:
回到排序的问题:
一、第一个和第二个的区别是什么?
第二个的opcode比第一个要少,反而比第一个要慢一些,因为构造函数多了参数,多了一个参数处理的opcode。参数处理是一个比较费时的操作,当我们在做代码优化时,一些不必要的参数能去掉就去掉;当一个函数有多个参数时,可以考虑通过一个数组将其封装后传递进来。
二、为啥第三个最慢?
因为其获取参数其本质上是一次对象成员方法的调用,方法的调用成本高于变量的获取
三、为啥第四个比第三个要快?
因为第四个的操作实质上获取变量,只不过其内部实现了魔术方法的调用,相对于用户定义的方法,内部函数的调用的效率会高。因此,当我们有一些PHP内核实现的方法可以调用时就不要重复发明轮子了。
四、为啥第四个比第二个要慢?
因为在PHP的对象获取变量的过程中,当成员变量在类的定义不在在时,会去调用PHP特有的魔术方法__get,多了一次魔术方法的调用。
总结一下:
原文地址:PHP成员变量获取对比, 感谢原作者分享。