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 相关文章推荐
php 变量定义方法
Jun 14 PHP
PHP中对用户身份认证实现两种方法
Jun 04 PHP
php设计模式 Chain Of Responsibility (职责链模式)
Jun 26 PHP
php cli 小技巧
Jun 03 PHP
php jquery 多文件上传简单实例
Dec 23 PHP
PHP调用.NET的WebService 简单实例
Mar 27 PHP
详解WordPress中过滤链接与过滤SQL语句的方法
Dec 18 PHP
java微信开发之上传下载多媒体文件
Jun 24 PHP
使用WAMP搭建PHP本地开发环境
May 10 PHP
微信开发之获取JSAPI TICKET
Jul 07 PHP
PHP魔术方法之__call与__callStatic使用方法
Jul 23 PHP
php常用日期时间函数实例小结
Jul 04 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
IIS+PHP+MySQL+Zend配置 (视频教程)
2006/12/13 PHP
第四章 php数学运算
2011/12/30 PHP
ThinkPHP中的关联模型注意点
2014/06/16 PHP
PHP+APACHE实现网址伪静态
2015/02/22 PHP
PHP+jQuery翻板抽奖功能实现
2015/10/19 PHP
详解WordPress中分类函数wp_list_categories的使用
2016/01/04 PHP
全面解读PHP的Yii框架中的日志功能
2016/03/17 PHP
PHP iconv()函数字符编码转换的问题讲解
2019/03/22 PHP
自动最大化窗口的Javascript代码
2013/05/22 Javascript
对Jquery中的ajax再封装,简化操作示例
2014/02/12 Javascript
jquery easyui使用心得
2014/07/07 Javascript
网站接入QQ登录的两种方法
2014/07/22 Javascript
常用的jQuery前端技巧收集
2014/12/24 Javascript
封装好的一个万能检测表单的方法
2015/01/21 Javascript
《JavaScript高级编程》学习笔记之object和array引用类型
2015/11/01 Javascript
js检测iframe是否加载完成的方法
2015/11/26 Javascript
bootstrap日期控件问题(双日期、清空等问题解决)
2017/04/19 Javascript
vue中appear的用法
2017/08/17 Javascript
浅谈Vue父子组件和非父子组件传值问题
2017/08/22 Javascript
VUE长按事件需求详解
2017/10/18 Javascript
javascript 构建模块化开发过程解析
2019/09/11 Javascript
JS中this的4种绑定规则详解
2020/02/04 Javascript
python开启多个子进程并行运行的方法
2015/04/18 Python
Python网络编程 Python套接字编程
2017/09/13 Python
python for和else语句趣谈
2019/07/02 Python
Django发送邮件和itsdangerous模块的配合使用解析
2019/08/10 Python
python自动提取文本中的时间(包含中文日期)
2020/08/31 Python
Urban Outfitters美国官网:美国生活方式品牌
2016/08/26 全球购物
Android interview questions
2016/12/25 面试题
中专生求职自荐信范文
2013/12/22 职场文书
2014年中职班主任工作总结
2014/12/16 职场文书
毕业实习证明范本
2015/06/16 职场文书
学生会任命书范本
2015/09/21 职场文书
2016年五四青年节校园广播稿
2015/12/17 职场文书
解析Redis Cluster原理
2021/06/21 Redis
Redis RDB技术底层原理详解
2021/09/04 Redis