日期传来消息关于PHP8JITRFC投票中,该议案得到绝大多数的支持,所以即将到来的PHP大版本PHP8中将会正式支持JIT。关于JIT可能大家都不陌生,Java的JVM和.net都是都有专门的支持。PHP从PHP7开始基于Opcache,HugePage,PGO等一些类JIT技术使得性能得到了极大改善,而另一个facebook开源的PHP虚拟机项目HHVM(HipHopVirtualMachine)则是完全采用了JIT技术来改善性能。那么JIT究竟是什么?为什么它有如此点石成金的本事呢?本文虫虫就和大家来一起了解下PHP8即将支持的JIT。
什么是JIT?
PHP实际上在底层也运行了一个虚拟机处理器——ZendVM。PHP将人类可读脚本编译成虚拟机理解的指令(我们称之为操作码),这个执行阶段就是我们所说的编译时。在运行时执行阶段,虚拟机(ZendVM)执行代码的指令(操作码)。
该体系一直运行很完美,还使用APC(php5.5以前)和OPCache(php7)来在编译时缓存代码的指令(opcode)。
JIT(Just-in-time),是一种编译器策略,采用字节码形式保存编译后的操作码,在运行时其转换为机器码即时执行。
在PHP中,这意味着JIT将为ZendVM生成的指令视为中间码,并生成依赖于体系结构的机器代码,所以代码执行不再需要ZendVM解释器,直接在CPU中执行。
PHPJIT实现细节
作为OPcache一个独立组件开发,可以在PHP内核编译时或运行时启用,生成的机器码,保存在Opcache共享区域;
使用LuaJIT的DynAsm库生成机器码,没有使用业界常用的LLVM;
目前只支持x86、x86_64POSIX平台,未来会提供更多平台的支持;
执行流程对比:
我们假设有个php代码chongchong.php执行:
没有Opcache缓存的执行过程:
chongchong.php-经过zend编译-opcode-PHP解释器-机器码
启用Opcache缓存的执行过程:
chongchong.php-查找opcache缓存,如果没有则进行zend编译为opcode并缓存-opcode-PHP解释器-机器码
JIT流程:
chongchong.php-编译-机器码
性能对比:
根据Mandelbrot基准测试的结果,在PHP7.4上启用JIT和未启用,针对下面脚本的性能对比,启用JIT后提高了4倍。
下面是各个语言都参与的基准测试结果如下:
为什么要引入JIT?
JIT会带来什么好处?会让我的网站更快吗?
在PHP7.0时候,PHP社区开始从Facebook的HHVM项目引入一些功能,将其包含在PHPNG补丁中,这功能大大改善了PHP利用内存和CPU核心的方式。
自PHP7.0开始,PHP的性能有了翻天覆地的改善,比如HashTable(PHP的核心数据结构)的优化,ZendVM中针对某些操作码的特化,针对特定序列的编译器优化以及对OPCache的持续改进。
虽然系能上有了很大的改善,php7.0比php5改善了70%,php7.3比7.0又改善了30%,但是后面呢?基本上php系能改善已经接近极限,除非架构上有重大的优化,那么这个架构优化是什么呢?目前来说就是JIT。
答案不是很确定,可能会让你的网站更快,但是不一定和明显。如果你的平台大量基于PHP实例,比如新浪这样的线上平台,PHPJIT会让你的服务器用的更少,更加节约硬件,更加省电,更加环保节能。
对个别网站来说,取决于你的网站瓶颈所在JIT不一定让你的网站更快,可能你网站首页有个2M的大图片,你把它改成K的图片带来性能改善可能更大。
I/O密集型和CPU密集型
当我们想要描述一段代码或应用程序的一般性能特征时,我们使用术语I/O密集型和CP密集型。比如:
如果我们可以改进(减少,优化)它正在进行的I/O,那么I/O密集型的代码会更快。
如果我们可以改进(减少,优化)CPU正在执行的指令,或者(神奇地)提高CPU的时钟速度,那么CPU密集型的代码会更快
一段代码或应用程序可以是I/O密集型,CPU密集型或CPU和I/O密集型。
一把来说,PHP应用程序(Web程序)往往受I/O限制,影响Web应用速度的主要原因是I/O问题:连接、读取和写入数据库,高速缓存。文件,Socket等。
CPU密集计算
CPU密集型可能大多数PHP码农都不很熟悉。因为对大多数PHP应用程序来说,一般都是连接到某个数据库,或者可能是数据库缓存,做一些字符处理,然后返回一个html/json/xml响应。
PHP执行实际实际上非常快,它是世界上解码速度最快的语言之一。ZendVM调用与I/O无关的函数的执行速度通常和在机器码中进行相同调用之间没有显著区别。显然存在差异,但事实是机器代码具有调用约定,ZendVM具有调用约定,机器码具有prologue,ZendVM也有prologue。是否在Zend操作码或机器码中调用some_c_level_function()对于应用程序的性能没有显著影响,尽管它似乎对该调用产生了重大影响。
JIT引入是不是为了使数值计算更快?
实际上虽然JIT的引入会大幅度提高数值计算的性能,但并不止如此。引入JIT后,会扩大PHP的使用范围,使得PHP成为一种通用型的语言,而不仅仅是个Web开发语言,尽管目前咋Web开发上面PHP做的确实是非常成功。
对富CPU密集型计算方面的改善,可以在机器学习,3D渲染,2D(图形)渲染和数据分析方面提高适用范围。