副标题#e#

 如何用 Rust 来为 PHP 助力

上周,一篇关于《PHP 兼 Zend 联合创始人 Zeev Suraski 宣布从 Zend 离职》的文章在中国程序员中炸开了锅。文章中的Zend创始人离职可谓引起轩然大波,各种对PHP不友好的声音也被传递了出来。虽然来自中国的PHP核心开发者鸟哥发声,但许多程序员对PHP的态度并不是很乐观。

作为我在Web领域的入门语言,对PHP的感情犹如初恋一般美好。但伴随着近几年来大家对PHP各种调侃、讽刺,加上新语言Go、Rust、Javascript等强势崛起。似乎让“我是一名PHP程序员”变得底气很不足。来源于Web、辉煌于Web、受限于Web,不得不说,这让脚本语言PHP陷入了困境,因为Web已经不再是那个靠渲染网页的PC时代,虽然目前有Swoole这样优秀的异步方案,但也很难让我对PHP提起足够的兴趣。

Rust作为近几年来逐步流行的新兴语言,凭借安全、高效、可靠,开始进入了人们的视野。周末,重拾起了PHP,想看看是否能够在PHP和Rust之间擦出一点点火花。当然,只是出于个人兴趣做的一些实验,这并不能为两个优秀的语言带来什么改变。但我想,至少能够让大家对PHP和Rust有更多的了解。

PHP7.4中的FFI

起初,PHP7.4中的新特性FFI引起我很大的兴趣,因为在很多语言中已经拥有了这样的特性,例如:Python、NodeJS、Ruby等,PHP首次的引入,让我觉得PHP团队正在朝着这方面进行努力。

我首先采用Rust编写了一个Fibonacci斐波那契数列,这里实现了一个简单的计算函数,我将他导出为动态链接库,以供PHP的FFI调用。

  1. #[no_mangle] 
  2. pub extern fn fib(n: i32) -> i32 { 
  3.   return match n { 
  4.     1 | 2 => 1, 
  5.     n => fib(n - 1) + fib(n - 2) 
  6.   } 

编写完成后,我尝试在PHP的FFI中进行调用,为了想对比下性能,我同时编写了一个PHP的函数:

  1. // 一个PHP的fib函数 
  2. function fib($n) { 
  3.     if ($n === 1 || $n === 2) { 
  4.         return 1; 
  5.     } else { 
  6.         return fib($n-1) + fib($n-2); 
  7.     }     

接下来,我在PHP中调用他们,为了能够看出性能差异,我将调用1000000次:

  1. // release模式 
  2. $ffiRelease = FFI::cdef( 
  3.     "int32_t fib(int32_t n);", 
  4.     "r2p-fib/target/release/libr2pfib.$libExtension"); 
  5.  
  6. $time_start = microtime(true); 
  7. for ($i=0; $i < 10000000; $i++) {  
  8.     $v = $ffiRelease->fib(12); 
  9.  
  10. echo '[Rust]Release执行时间:' . (microtime(true) - $time_start).PHP_EOL; 

从测试结果来看,Rust的FFI结果是让人惊喜的。

PHP的计算耗时30秒以上,Rust仅仅用了6秒。

当我为此欣喜若狂的时候,我又尝试了下PHP的FFI调用生成字符串,在PHP中是类似这样一个方法:

  1. function text_generate($num) { 
  2.     $result = "💣"; 
  3.     $result .= str_repeat("na ",$num); 
  4.     $result .= "Batman! 💣"; 
  5.     return $result; 

结果Rust由于在PHP的FFI中间字符串转换的损耗,性能并没有达到预想那样。

PHP扩展调用Rust动态库

因为第一个操作,让我想到了FFI在多次调用性能损耗是很大的,这时我想实现在PHP扩展中来调用Rust动态库。

同时为了对比,我编写一个C的Fib函数进行比较。

我创建了一个名为rust的PHP扩展,完成了关于我们上面编写的Rust函数的调用。

  1. ZEND_BEGIN_ARG_INFO(arginfo_rust_fib, 0) 
  2.     ZEND_ARG_INFO(0, number) 
  3. ZEND_END_ARG_INFO() 
  4.  
  5. /* {{{ int rust_fib( [ int $var ] ) 
  6.  */ 
  7. PHP_FUNCTION(rust_fib) 
  8.     zend_long number = 0; 
  9.     zend_long result = 0; 
  10.     ZEND_PARSE_PARAMETERS_START(0, 1) 
  11.         Z_PARAM_OPTIONAL 
  12.         Z_PARAM_LONG(number) 
  13.     ZEND_PARSE_PARAMETERS_END(); 
  14.  
  15.     if (number == 0) { 
  16.         RETURN_LONG(result); 
  17.     } else { 
  18.         result = fib(number); 
  19.         RETURN_LONG(result); 
  20.     } 
  21. /* }}}*/ 

结果同样让我惊喜,没有了FFI,它确实提升了20%左右的性能。但很明显,它的实现复杂度更高了。

采用Rust编写PHP扩展

将PHP的Zend API结构导出,直接在Rust中实现PHP模块的编写。虽然自己没有足够的精力去做这样的尝试,但我觉得这是一个可行的方法。我想在未来的时间内完成这样的尝试。

总结

#p#副标题#e#

虽然影响PHP更好的发展的主要因素并不是性能,尽管本篇只是从性能提升入手对PHP进行了测试,但我觉得,作为一个拥有如此庞大开发群体的Web语言,也是时候需要跟Go、Rust、Node这些新兴语言学习了,建立并不局限于Web领域的新的生态体系,大胆的去做一些新的尝试,让PHP不再是“世界第一语言”。

本文完整测试代码及结果请参考:https://github.com/llgoer/php-ffi-rust/

【编辑推荐】

  1. 程序员们用了就离不开的10款开发软件,今天全部安利给大家
  2. 比较全面的C#帮助类,各种功能性代码,可以直接拿来即用
  3. PHP 兼 Zend 联合创始人 Zeev Suraski 宣布从 Zend 离职
  4. 适用于Vue开发人员的原型工具OverVue
  5. 利用深度学习自动补全 Python 代码,这个开源项目有点牛

【责任编辑:张燕妮 TEL:(010)68476606】
点赞 0

dawei

【声明】:北京站长网内容转载自互联网,其相关言论仅代表作者个人观点绝非权威,不代表本站立场。如您发现内容存在版权问题,请提交相关链接至邮箱:bqsm@foxmail.com,我们将及时予以处理。