我们为什么要迁移PHP到HHVM

程序员日志 · 2014-12-26 18:33

简介


该调研是2013年10月份做的,目标是寻找更好的PHP引擎,来代替百度各产品线正在使用的PHP 5.2。

环境说明


机器环境:

  • cpu: Intel(R) Xeon(R) CPU E5-2620 0 @ 2.00GHz, 12核。

  • 内存:64G

引擎:

  • php 5.2.17 (当时百度所用版本)

  • php 5.5.4 (当时最新版本)

  • hhvm 2.3-dev (当时最新版本)

为了公平起见,三个引擎都采用-O2的编译优化选项。

加速器:

  • php5.2.17:采用eaccelerator

  • php5.5.4:eacc不可用,采用自带的zend opcache

  • hhvm:内置opcode缓存,开启jit

扩展:

由于目前的百度扩展只有很少移植到了hhvm,所以为了公平起见,其它百度扩展也在php5.2和php5.5中禁用,而采用纯php的方式实现,实践证明采用纯php实现性能并未有太大损失。

性能


下面用了不同的场景,来比较PHP引擎的性能。

场景一:纯cpu计算

用php官方提供的测试脚本:

  • bench.php : https://github.com/php/php-src/blob/master/Zend/bench.php

  • micro_bench.php : https://github.com/php/php-src/blob/master/Zend/micro_bench.php

  • 第三方提供的测试脚本bench_third.php : http://www.php-benchmark-script.com/

测试结果如下:

引擎 bench.php 耗时 micro_bench.php 耗时 bench_third.php 耗时
php5.2 6.692s 41.890s 9.226s
php5.5 3.609s 14.972s 5.893s
hhvm 0.579s 5.832s 2.869s

分析:php历次新版本对性能都有优化,因此php5.5在纯cpu计算上比php5.2有明显优势。而hhvm采用jit机制,将php转化为本地代码执行,所以性能上比执行opcode的php5.5还要快上很多。

结论:对于纯cpu型计算,hhvm>>php5.5>>php5.2。

场景二:简单http服务

简单的http服务即由一个简单的php逻辑接口层加一个后端C模块(或通用存储)构成的服务,这里用了百度内部一个实际的简单服务为例。

简单做了一下性能测试,client并发10,client与service同机部署,service与后端C模块不同机部署。结果如下:

引擎 平均响应时间 Cpu idle Qps
php5.2 3.6ms 50% 1900
php5.5 2.9ms 60% 2500
hhvm 2.0ms 60% 3900

分析:对于简单http服务,自身也有一部分的逻辑计算,除此之外,hhvm是webserver与php引擎一体化,不需要 webserver->php这一层的开销,因此在响应时间和qps上有更明显的优势。由于C模块后端本身响应时间较短(1ms),所以不同引擎的 整体响应时间的区别就比较明显。

结论:简单http服务,在qps上hhvm>>php5.5>php5.2,在响应时间上,除掉后端服务耗时,也是hhvm>>php5.5>php5.2。

场景三:产品前端UI

这里用了贴吧的最新版ui,串行访问后端。

采用20个并发,测试结果如下:

引擎 整体耗时 除访问后端之外的耗时 idle qps
php5.2 520ms 127ms 50% 40
php5.5 500ms 107ms 45% 45
hhvm 470ms 72ms 65% 50

分析:ui大部分时间消耗在后端服务访问,这方面引擎不起作用。而另一方面,除后端服务外的耗时也很高,这部分对cpu消耗较大,导致qps较低。

显然,这个效果并不是很符合我们的预期,为了达到理想的qps,我们需要想方法进一步降低除后端之外的耗时。

首先我们在PHP5.2下运行xhprof,找出了一些较耗时的函数,并进行了优化:

  1. 去掉不必要的编码转换,减少15ms

  2. 去掉不必要的warning日志,耗时减小4.3ms

  3. 改进xss转码函数实现,耗时减少1.8ms

通过上面几个优化,HHVM下ui的耗时降到了52ms,其中模板耗时34ms。但是这样仍不令人满意,如何进一步优化呢?

既然上面的xhprof是在PHP5.2下执行的,那我们尝试直接在HHVM下运行xhprof会如何?结果发现,HHVM下xhprof的结果与PHP5.2下有很大不同,发现了一些新的优化点,进行了优化:

  1. 有好几个函数会在每个请求中重复计算同样的信息,加上apc cache之后,耗时减小16ms

  2. 用户自定义的errorHandler,很耗时去掉后耗时减小7ms

  3. 去掉一些debug日志,耗时减小1ms

通过上面的优化,我们将HHVM下ui的耗时降到了26ms。那么,同样的优化,作用在PHP5.2或PHP5.5下又如何呢?

引擎 优化前 业务优化后
php5.2 127ms 103ms
php5.5 107ms 87ms
hhvm 72ms 26ms

令人失望。。。究其原因,是因为在php下耗时在各函数的分布相当分散,即使优化个别函数,也难以有大辐提升。

而在hhvm下,由于引擎本身性能较高,因此一些原来实现得不合理而比较耗时的函数就突显出来了,因此针对这些函数进行优化,可以取得更加明显的效果。

结论:对于产品前端UI,引擎可以在一定程度上优化响应时间和qps,hhvm>php5.5>php5.2,如再结合一些针对hhvm的优化,可以大幅提升hhvm下的qps。

兼容性


语言兼容性


语言方面的兼容性来自两个方面:

php不同版本的语法差异

hhvm是基于php5.4的语法标准,因此,hhvm和php5.5与我们目前用的php5.2都存在语法差异,php5.5差异更大一些。

目前所遇到的差异个数如下:

引擎 影响处理结果 不影响结果,打warning
php5.5 1个 1个
hhvm 1个 1个

所有差异可见这里:

  • http://www.php.net/manual/en/migration53.php

  • http://www.php.net/manual/en/migration54.php

  • http://www.php.net/manual/en/migration55.php

hhvm与zend引擎之间的实现差异

由于是两套不同实现,因此hhvm与zend引擎也有差异。

目前遇到的差异数如下:

影响处理结果 不影响结果,打warning
2个 0个

hhvm文档也提到了一些差异:https://github.com/facebook/hhvm/blob/master/hphp/doc/inconsistencies

结论:总体上,hhvm与php5.5在语言方面与php5.2都存在一些差异,并且对处理结果造成影响。不过,大部分的差异都会导致执行时报fatal或warning,我们只需根据相应提示修改代码即可,改动成本不大。

扩展兼容性


php5.5

将原百度用到的扩展在php5.5下编译,大部分能编译通过,少数编译失败:

  1. 升级到最新版本可解决,5个

  2. 修改代码解决,2个

hhvm

hhvm内置了很多常用的第三方扩展,因此对于第三方扩展,基本无须迁移。主要需要考虑的是百度扩展的迁移。

hhvm编写扩展有两种方式:

一、静态编译

  1. 编写扩展的idl文件(json格式)

  2. 根据idl自动生成代码框架

  3. 填充代码框架,实现自己的逻辑

  4. 重新编译hhvm

缺点:

  1. 现有百度扩展需要重新改造,与获取输入参数、返回结果相关及其它调用到php.h中的api的地方都需要重写。这是主要的成本。

  2. hhvm的api目前还不太稳定,2.1.0下编写的扩展,在2.2.0下就编译不通过了。。

优点:

  1. 新扩展编写,比在zend下编写更加方便,hhvm提供类似于boost variant的api,处理输入输出很方便

二、动态加载(HNI:hhvm native interface)

  1. 编写函数声明.php(语法与php很像)

  2. 编写函数实现.cpp

  3. hphpize && cmake . && make,只须单独编译扩展

动态加载更为方便,因此推荐使用这种方式编写扩展。

三、ZendCompatLayer

hhvm正在实现一个ZendCompatLayer,即兼容zend扩展的一个层,实现zend引擎相关api,以方便现有扩展迁移到hhvm。具体见:https://github.com/facebook/hhvm/blob/master/hphp/doc/php.extension.compat.layer

按照其描述,zend扩展可以不做太多修改,即可迁移到hhvm。

  1. 尝试开启ZendCompatLayer功能,并编译包中自带的yaml扩展,确认ZendCompatLayer已经可用

  2. 尝试使用ZendCompatLayer编译两个百度扩展,发现很多问题,比如链接时缺少各种函数、加载不了ini、调用不了minit/rinit等函 数、gcc版本不兼容、原扩展不支持多线程环境等等,经过各种修改hhvm和扩展代码,终于跑通,但成本很高,而且稳定性很难保障。

  3. 测试了一下ZendCompatLayer的性能,用一个包含string, int和嵌套数组的数组,反复调用一个百度扩展做打包解包,结果如下:

耗时 实现方式
199 ms HNI
769 ms HNI+ZendCompat
348 ms php5.2

结论:ZendCompatLayer目前还不完善,很多api未实现,另外性能还较差,对性能要求高的扩展还是建议用hhvm原生方式实现,而长尾扩展可以用ZendCompatLayer,减少迁移与维护成本。

结论:php5.5对扩展保持较好的兼容性,迁移成本不高。hhvm目前扩展迁移成本较高,不过如果有ZendCompatLayer,也能大大减小迁移成本,可以等ZendCompatLayer完善之后,再做进一步调研。

系统兼容性


php5.5对操作系统基本没有什么要求。

hhvm对系统要求较高,需要在ubuntu/centos下编译、运行,百度大部分机器不符合条件。。

不过,经过研究后发现,可以通过打包so方式,让hhvm能在百度机器运行,上面的测试都是在百度机器上进行的,没有发现什么问题。

根据同样的原理,通过打包依赖so和头文件,也能让hhvm在百度机器上编译,已经成功编译。

内核方面,hhvm要求至少2.6.32系统,百度大部分系统都符合要求。

结论:系统兼容性上,php5.5和hhvm都没有什么问题。

配置兼容性


php5.5对php.ini保持兼容,但php-fpm.conf则变成ini格式,原来的php-fpm.conf需要修改,并且php-fpm也被编译成一个单独的可执行文件,而不再和php-cgi共用。

hhvm的配置是一种新的hdf格式,而且其配置项都不相同,不一定每个php.ini配置都能找到对应的配置项,不过常见的都有。新版hhvm也支持指定php.ini文件了,不过仍然不能支持所有配置项。

hhvm是webserver与php一体化,webserver的常见功能,如rewrite、简单防攻击、proxy、日志、静态文件服务也都 有。因此webserver的相关配置也要成hhvm的配置格式。不过现在HHVM也支持fastcgi模式了,所以这个已经不成问题。

结论:配置兼容性上php5.5更好一些。不过配置文件也并不多,人工改造成本也不是很高,因此这方面问题并不大。

新功能


php从5.3到5.5增加了很多新功能,具体可以见:

  • http://www.php.net/manual/en/migration53.php

  • http://www.php.net/manual/en/migration54.php

  • http://www.php.net/manual/en/migration55.php

总体上可概括为语法更灵活,功能更丰富。

hhvm的语法基于5.4,库的支持也不如php5.5丰富。但hhvm提供了几个功能,可能对性能优化会有很大帮助:

  1. StartupDocument:可以在启动时执行该文件,在此文件中做一些全局初始化工作,并将结果变量通过apc传递给worker线程。这样我们将一些重复的初始化工作放到该文件,避免每个请求都执行同样的工作。

  2. ThreadDocuments:可以用后台线程执行一个文件,做一些比如信息收集、定时更新之类的操作,避免在每个请求中执行相应过程,影响在线请求的响应时间。

  3. Pagelet Server/Xbox Server:可以启动一个子Server,子Server维护一个线程池,主Server可以将一些计算任务分发给子Server异步执行,从而实现并 行化。我尝试了一下这个功能,利用xbox server很方便地就实现了并行调用后端服务的功能,确实很方便,而且这个机制是通用的,不仅后端调用可以并行,本地计算也可以并行,可以考虑将模板并 行渲染?呵呵。

  4. 多线程架构:这其实不算一个功能,而是hhvm多线程架构给我们带来的天然便利。因为在同一个进程内,内存共享会很方便,访问速度更快,而像一些比如连接池的功能,也变得可以实现。

上述功能的介绍详见:

  • https://github.com/facebook/hhvm/blob/master/hphp/doc/server.documents

  • https://github.com/facebook/hhvm/blob/master/hphp/doc/threading

开发者支持


php5.5的开发者支持很好,文档全,api非常稳定。

相比之下,hhvm的开发者支持目前还比较弱,文档很少,很多地方要自己摸索,api还不太稳定,不过hhvm的社区很活跃,github有很多人在提issue和patch,hhvm官方团队也很给力,迭代速度很快,每8周发布一个稳定版本。

总体结论


总体上,使用php5.5,性能会有一定提升,迁移成本较小。

使用hhvm,性能会有很大提升,但迁移成本会较大,主要是扩展方面的迁移成本。


UPDATE

看这里的结论很明显,百度大部分产品线使用了HHVM,因为我们相信资源的节约对于 我们这么大规模的应用收益很大。

读到这里,你可能会有疑问思考你到底要不要升级到HHVM?你想知道PHP7对比怎么样?

这里是你需要迁移HHVM的原因:

  1. 你现在服务器成本比较大,想要节约成本,或者流量持续增长资源不足。

  2. 你需要比较透明的提升性能

  3. 你没有什么自有扩展,或者有时间迁移扩展(扩展迁移到HHVM的成本很低,后面会做相关的介绍)

  4. 你想使用Hack语言

  5. 或者你们想尝试新的技术

同样,这也有不建议迁移HHVM的原因:

  1. 你的服务器及流量很少

  2. 你的业务很稳定,没有性能上的要求

  3. 你们是土豪,机器随便堆,预算一大把

这还有一些答疑:

  • Q: 有人提到运维成本的问题,HHVM真的不容易运维么?

  • A: HHVM的多线程模型是相对PHP-FPM不稳定,但是有非常多的进程守护解决方案可以很好的解决, 这个不是个问题,以百度现在的运维来看比较稳定。

  • Q:PHP7的性能不是已经不错了么?为什么还要迁移到HHVM?

  • A:首先PHP7还处于开发状态,首先迁移大版本的风险还是有的,至少现在考虑是不靠谱的,如果愿意等, 可以等到PHP7出来以后来进行对比。看看哪个更适合你,目前这个阶段HHVM是比PHP更优的选择。

做你们的测试(Do Your Own Benchmarking)

如果你们决定迁移了,那么开始前也建议做一次你们业务相关的评测,以确保收益在预期范围内。

我们为什么要迁移PHP到HHVM的更多相关文章

  1. 转: HHVM at Baidu

    评注: 一个项目迁移的问题考虑与实现使用.非常之详细. 转:http://lamp.baidu.com/2014/11/04/hhvm-in-baidu/ 在这之前我们介绍了我们为什么要迁移PHP到H ...

  2. HHVM 是如何提升 PHP 性能的?

    背景 HHVM 是 Facebook 开发的高性能 PHP 虚拟机,宣称比官方的快9倍,我很好奇,于是抽空简单了解了一下,并整理出这篇文章,希望能回答清楚两方面的问题: HHVM 到底靠谱么?是否可以 ...

  3. 【问底】徐汉彬:PHP7和HHVM的性能之争 (真是学到了很多)

    来源:http://www.csdn.net/article/2014-12-25/2823234 作者:徐汉彬 摘要:近日,PHP7和HHVM的性能之争成为了一个讨论热点,但毫无疑问,它们都在提升P ...

  4. 【问底】徐汉彬:PHP7和HHVM的性能之争

    本文来源于:  http://www.csdn.net/article/2014-12-25/2823234 [导读]徐汉彬曾在阿里巴巴和腾讯从事4年多的技术研发工作,负责过日请求量过亿的Web系统升 ...

  5. 基于本地存储的kvm虚拟机在线迁移

    基于本地存储的kvm虚拟机在线迁移 kvm虚拟机迁移分为4种(1)热迁移基于共享存储(2)热迁移基于本地存储(3)冷迁移基于共享存储(4)冷迁移基于本地存储 这里介绍的是基于本地存储的热迁移 动态块迁 ...

  6. 【SQLServer】记一次数据迁移-标识重复的简单处理

    汇总篇:http://www.cnblogs.com/dunitian/p/4822808.html#tsql 今天在数据迁移的时候因为手贱遇到一个坑爹问题,发来大家乐乐,也传授新手点经验 迁移惯用就 ...

  7. MVC5+EF6+MYSQl,使用codeFirst的数据迁移

    之前本人在用MVC4+EF5+MYSQL搭建自己的博客.地址:www.seesharply.com;遇到一个问题,就是采用ef的codefirst模式来编写程序,我们一般会在程序开发初期直接在glob ...

  8. 烂泥:wiki系统confluence5.6.6安装、中文、破解及迁移

    本文由ilanniweb提供友情赞助,首发于烂泥行天下 想要获得更多的文章,可以关注我的微信ilanniweb confluence是一个专业的企业知识管理与协同软件,可以用于构建企业wiki.通过它 ...

  9. ASP.NET Aries 2.0 发布(原来的源码SVN已关闭,开源源码已迁移到GitHub)

    主要更新: 1:增加子目录部署支持. 2:增加Taurus.MVC支持. 3:优化及Bug修复. 1:增加子目录部署支持: 其实在重写Aries框架的时候,我是去掉了目录部署功能的,主要是为了加快Ar ...

随机推荐

  1. 环境变量的配置-java-JMETER - 【Linux】

    rz上传 lz下载 步骤: . Linux下首先安装Jdk: . 下载apache-jmeter-4.0.tgz,复制到Linux系统中的/opt目录下: . 解压apache-jmeter-4.0. ...

  2. centos7.6 安装nginx-1.14.2

    一.安装所需依赖环境 yum -y install gcc-c++ pcre pcre-devel zlib zlib-devel openssl openssl-devel 二.下载nginx官方源 ...

  3. 一文拆解Faas的真实案例

    欢迎大家前往腾讯云+社区,获取更多腾讯海量技术实践干货哦~ 本文来自腾讯云技术沙龙,本次沙龙主题为Serverless架构开发与SCF部署实践 刘敏洁:具有多年云计算行业经验,曾任职于华为.UClou ...

  4. uiimageview 的 animation 动画

    NSMutableArray *meiArr = [NSMutableArray arrayWithCapacity:4]; for (int i = 0; i < 4; i++) { NSSt ...

  5. Redis Jedis简介

    Redis是一种基于内存类型的数据存储工具 Jedis是一个用java写的Redis数据库操作的客户端,通过Jedis,可以很方便的对redis数据库进行操作.Jedis通过Jedis Pool进行R ...

  6. 前端_CSS

    目录 CSS语法 CSS的四种引入方式 CSS选择器 CSS属性操作 补充 示例(一些小模板) CSS语法 CSS 规则由两个主要的部分构成:选择器,以及一条或多条声明. 1 2 3 4 5 6 7 ...

  7. Servlet 3.0 对异步处理的支持

    Servlet 3.0 实现了对异步处理的支持 通过利用注解@WebServlet(urlPatterns="/AServlet" AysnsSupported=true) 让后n ...

  8. php----函数大全

    字符串函数 数组函数 数学函数

  9. shiro+springmvc 都使用缓存

    基于涛哥shiro案例16 的这时候要配置service方法的缓存 在spring-config.xml添加 <context:annotation-config /> <cache ...

  10. 剑指offer :从尾到头打印链表

    题目描述: 输入一个链表,按链表值从尾到头的顺序返回一个ArrayList. 解题思路: 链表的遍历只能从头向尾进行,要从尾到头输出,考虑用栈.先从头到尾遍历一次链表,同时将值进栈,再清空栈,同时将值 ...