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 相关文章推荐
默默简单的写了一个模板引擎
Jan 02 PHP
PHP Memcached应用实现代码
Feb 08 PHP
php操作SVN版本服务器类代码
Nov 27 PHP
php.ini-dist 和 php.ini-recommended 的区别介绍(方便开发与安全的朋友)
Jul 01 PHP
php_screw安装使用教程(另一个PHP代码加密实现)
May 29 PHP
Windows下安装PHP单元测试环境PHPUnit图文教程
Oct 24 PHP
PHP模糊查询的实现方法(推荐)
Sep 06 PHP
PHP实现验证码校验功能
Nov 16 PHP
在Laravel5.6中使用Swoole的协程数据库查询
Jun 15 PHP
PHP正则判断一个变量是否为正整数的方法
Feb 27 PHP
laravel 根据不同组织加载不同视图的实现
Oct 14 PHP
php 使用expat方式解析xml文件操作示例
Nov 26 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从FLV文件获取视频预览图的方法
2015/03/12 PHP
Zend Framework框架教程之Zend_Db_Table_Rowset用法实例分析
2016/03/21 PHP
Paypal实现循环扣款(订阅)功能
2017/03/23 PHP
利用腾讯的ip地址库做ip物理地址定位
2010/07/24 Javascript
基于jQuery的自动完成插件
2011/02/03 Javascript
js图片滚动效果时间可随意设定当鼠标移上去时停止
2014/06/26 Javascript
详解jQuery中ajax.load()方法
2017/01/25 Javascript
H5图片压缩与上传实例
2017/04/21 Javascript
NodeJS父进程与子进程资源共享原理与实现方法
2018/03/16 NodeJs
详解如何在vue项目中使用lodop打印插件
2018/09/27 Javascript
读懂CommonJS的模块加载
2019/04/19 Javascript
JavaScript静态作用域和动态作用域实例详解
2019/06/17 Javascript
JS表单验证插件之数据与逻辑分离操作实例分析【策略模式】
2020/05/01 Javascript
vue自定义组件实现双向绑定
2021/01/13 Vue.js
[02:41]《西雅图我们来了》2015国际邀请赛出征全记录
2015/07/23 DOTA
Python装饰器decorator用法实例
2014/11/10 Python
python实现统计代码行数的方法
2015/05/22 Python
Python操作mongodb的9个步骤
2018/06/04 Python
Python中的十大图像处理工具(小结)
2019/06/10 Python
用Python抢火车票的简单小程序实现解析
2019/08/14 Python
Django数据结果集序列化并展示实现过程
2020/04/22 Python
django使用graphql的实例
2020/09/02 Python
HTML5时代CSS设置漂亮字体取代图片
2014/09/04 HTML / CSS
美国领先的礼品卡网站:GiftCards.com
2016/11/02 全球购物
美国电力供应商店/电气批发商:USESI
2018/10/12 全球购物
什么是规则表达式
2012/05/03 面试题
行政助理岗位职责范文
2013/12/03 职场文书
爷爷追悼会答谢词
2014/01/24 职场文书
工作目标责任书
2014/07/23 职场文书
2014收银员工作总结范文
2014/12/16 职场文书
医药公司开票员岗位职责
2015/04/15 职场文书
2015年教务工作总结
2015/05/23 职场文书
保护地球的宣传语
2015/07/13 职场文书
Django展示可视化图表的多种方式
2021/04/08 Python
Python 语言实现六大查找算法
2021/06/30 Python
MySQL的全局锁和表级锁的具体使用
2021/08/23 MySQL