PHP代码优化之成员变量获取速度对比


Posted in PHP onFebruary 28, 2014

有如下4个代码示例,你认为他们创建对象,并且获得成员变量的速度排序是怎样的?

1:将成员变量设置为public,通过赋值操作给成员变量赋值,直接获取变量

<?php
class Foo {
    public $id;
}
$data = new Foo;
$data->id = 10;
echo $data->id;
?>

2:将成员变量设置为public,通过构造函数设置成员变量的值,直接获取变量
<?php
class Foo2 {
 public $id;
 public function __construct($id) {
  $this->id = $id;
 }
}
$data = new Foo2(10);
echo $data->id;
?>

3:将成员变量设置为protected,通过构造函数设置成员变量的值,通过魔术方法获取变量
<?php
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,通过构造函数设置成员变量的值,通过成员方法获取变量
<?php
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内核创建对象的过程分为三步:

ZEND_FETCH_CLASS 根据类名获取存储类的变量,其实现为一个hashtalbe EG(class_table) 的查找操作
NEW 初始化对象,将EX(call)->fbc指向构造函数指针。
调用构造函数,其调用和其它的函数调用是一样,都是调用zend_do_fcall_common_helper_SPEC

二、魔术方法的调用是通过条件触发的,并不是直接调用,如我们示例中的成员变量id的获取

(zend_std_read_property),其步骤为:
获取对象的属性,如果存在,转第二步;如果没有相关属性,转第三步
从对象的properties查找是否存在与名称对应的属性存在,如果存在返回结果,如果不存在,转第三步
如果存在__get魔术方法,则调用此方法获取变量,如果不存在,报错
回到排序的问题:

一、第一个和第二个的区别是什么?

第二个的opcode比第一个要少,反而比第一个要慢一些,因为构造函数多了参数,多了一个参数处理的opcode。参数处理是一个比较费时的操作,当我们在做代码优化时,一些不必要的参数能去掉就去掉;当一个函数有多个参数时,可以考虑通过一个数组将其封装后传递进来。

二、为啥第三个最慢?

因为其获取参数其本质上是一次对象成员方法的调用,方法的调用成本高于变量的获取

三、为啥第四个比第三个要快?

因为第四个的操作实质上获取变量,只不过其内部实现了魔术方法的调用,相对于用户定义的方法,内部函数的调用的效率会高。因此,当我们有一些PHP内核实现的方法可以调用时就不要重复发明轮子了。
四、为啥第四个比第二个要慢?
因为在PHP的对象获取变量的过程中,当成员变量在类的定义不在在时,会去调用PHP特有的魔术方法__get,多了一次魔术方法的调用。

总结一下:

1.使用PHP内置函数
2.并不是事必面向对象(OOP),面向对象往往开销很大,每个方法和对象调用都会消耗很多内存。
3.尽量少用魔术方法 -- 除非有必要,不要用框架,因为框架都有大量的魔术方法使用。
4.在性能优先的应用场景中,将成员变量不失为一种比较好的方法,当你需要用到OOP时。
5.能使用PHP语法结构的不要用函数,能使用内置函数的不要自己写,能用函数的不要用对象

PHP 相关文章推荐
比file_get_contents稳定的curl_get_contents分享
Jan 11 PHP
ThinkPHP表单自动验证实例
Oct 13 PHP
php一行代码获取文件后缀名实例分析
Nov 12 PHP
PHP计算百度地图两个GPS坐标之间距离的方法
Jan 09 PHP
PHP获取数组最后一个值的2种方法
Jan 21 PHP
php微信支付之APP支付方法
Mar 04 PHP
php常用表单验证类用法实例
Jun 18 PHP
PHP模拟asp.net的StringBuilder类实现方法
Aug 08 PHP
yii2中添加验证码的实现方法
Jan 09 PHP
CI框架常用函数封装实例
Nov 21 PHP
laravel5使用freetds连接sql server的方法
Dec 07 PHP
thinkphp3.2框架中where条件查询用法总结
Aug 13 PHP
php递归方法实现无限分类实例代码
Feb 28 #PHP
PHP中常用的转义函数
Feb 28 #PHP
PHP中鲜为人知的10个函数
Feb 28 #PHP
php中调用其他系统http接口的方法说明
Feb 28 #PHP
PHP URL参数获取方式的四种例子
Feb 28 #PHP
php使用多个进程同时控制文件读写示例
Feb 28 #PHP
php使用curl抓取qq空间的访客信息示例
Feb 28 #PHP
You might like
php下将图片以二进制存入mysql数据库中并显示的实现代码
2010/05/27 PHP
php源代码安装常见错误与解决办法分享
2013/05/28 PHP
Linux环境下搭建php开发环境的操作步骤
2013/06/17 PHP
编译php 5.2.14+fpm+memcached(具体操作详解)
2013/06/18 PHP
php之curl设置超时实例
2014/11/03 PHP
php批量删除超链接的实现方法
2015/10/19 PHP
WordPress中用于检索模版的相关PHP函数使用解析
2015/12/15 PHP
一些易混淆且不常用的属性,希望有用
2007/01/29 Javascript
CLASS_CONFUSION JS混淆 全源码
2007/12/12 Javascript
JavaScript 对话框和状态栏使用说明
2009/10/25 Javascript
40款非常棒的jQuery 插件和制作教程(系列一)
2011/10/26 Javascript
jquery乱码与contentType属性设置问题解决方案
2013/01/07 Javascript
jquery延迟加载外部js实现代码
2013/01/11 Javascript
window.requestAnimationFrame是什么意思,怎么用
2013/01/13 Javascript
node.js开机自启动脚本文件
2014/12/24 Javascript
Jquery判断radio、selelct、checkbox是否选中及获取选中值方法总结
2015/04/15 Javascript
Bootstrap3制作图片轮播效果
2016/05/12 Javascript
Map.vue基于百度地图组件重构笔记分享
2017/04/17 Javascript
利用C/C++编写node.js原生模块的方法教程
2017/07/07 Javascript
nodejs接入阿里大鱼短信验证码的方法
2017/07/10 NodeJs
Vue filter介绍及详细使用
2018/04/04 Javascript
vue router 源码概览案例分析
2018/10/09 Javascript
解决小程序无法触发SESSION问题
2020/02/03 Javascript
原生JavaScript写出Tabs标签页的实例代码
2020/07/20 Javascript
[17:13]DOTA2 HEROS教学视频教你分分钟做大人-斯拉克
2014/06/13 DOTA
python使用socket进行简单网络连接的方法
2015/04/29 Python
python实现TF-IDF算法解析
2018/01/02 Python
TensorFlow如何实现反向传播
2018/02/06 Python
Python帮你识破双11的套路
2019/11/11 Python
PyCharm MySQL可视化Database配置过程图解
2020/06/09 Python
详解快速开发基于 HTML5 网络拓扑图应用
2018/01/08 HTML / CSS
企业党员个人自我评价
2014/09/20 职场文书
民主生活会批评与自我批评总结
2014/10/17 职场文书
运动会加油稿50字
2015/07/21 职场文书
特种设备安全管理制度
2015/08/06 职场文书
Pytorch 实现变量类型转换
2021/05/17 Python