内容来源:2018 年 5 月 20 日,腾讯企点开放平台技能卖力人熊月在“PHPCon China 2018 技能峰会”进行《嵌入式PHP的探索实践》演讲分享。
IT 大咖说作为独家视频互助方,经主理方和讲者审阅授权发布。

阅读字数:2410 | 7分钟阅读

获取高朋演讲视频及PPT,请复制:http://t.cn/RDsTba0,粘贴至浏览器即可。

php内嵌c摸索嵌入式PHP与C/C联合的无穷种可能 Docker

择要

对付拥有很多繁芜业务场景的tob领域,“开拓效率”和“性能”常常是我们考虑的两个紧张问题,PHP作为脚本措辞,本身适用于快速开拓业务逻辑,同时为理解决PHP特定的性能瓶颈,一样平常将C++/PHP结合,在PHP代码里调用C/C++扩展。
这次我们带来了不同思路的探索:将php嵌入到高性能C/C++框架运行,将C/C++框架作为容器,完美结合php快速开拓上风和C/C++高性能特点。
Zend Engine供应了一种嵌入式开拓模式,我们利用这一特性使它可以在C/C++的环境中单独实行PHP脚本,并且支持多实例运行,可以在C/C++协程框架中运行。
嵌入式PHP也为在任意C/C++协程框架结合带来无限可能,包括在C++的客户端上运行PHP。

嵌入式PHP

作为一门后台开拓措辞PHP有着不同的发展阶段。
最开始是大家都比较熟习的LAMP,接着是PHP-fpm和fastcgi,再今后是swoole,之后在swoole的根本上又新增了协程。

为了便于理解,在先容嵌入式PHP之前要先讲下SAPI的观点。
SAPI即后台运用程序编程接口,是PHP与其他运用程序交互的接口。
常见的SAPI有cgi、fpm、cli、Apace2 hander,而嵌入式PHP(embed)也是个中一类。

业务场景

我们最初的业务框架是基于TSF2.0,底层为Zend Engine和扩展,扩展最核心的部分是基于swoole。
在此之上是TSF PHP层,包含协程调度器、微做事框架、监控管理进程、MVC模式。
最上层才是真正的实行逻辑的PHP脚本。

这样一套框架存在几个问题。
PHP原生的Generator协程须要合营yield利用,对开拓来说不怎么友好,由于yield的利用机遇不太好确定,尤其是对付新手。
由于协程调度器是用原生PHP实现的,因此相对其他措辞在性能上会差些,特殊是在高并发场景下。
还有便是低版本swoole不足稳定,问题最多的便是在内存透露这块。
大家都知道腾讯内部有很多公用组件,这些组件接口大多是用C++实现的,为此我们须要做的是用扩展对接PHP与其他组件,而问题就在于扩展无法利用上层的PHP协程。

方案:SPP+PHP

为什么选择嵌入式PHP

SNG中有个非常有名的C++后台框架SPP,它是一个高性能的网络框架,起始于2008年,被广泛的运用于SNG的各个业务线。
众所周知开拓效率一贯是PHP的长处,性能方面则是短板。
以是我们就在想能不能将SPP和PHP结合起来兼顾高性能和开拓效率,嵌入式PHP无疑是很好的结合方案。

SPP紧张有5个模块。
proxy用来处理新要求,内存行列步队是proxy和workgroup交互的内存区域,workgroup是和后台逻辑脚本交互的模块,controller作为掌握核心来掌握proxy和workgroup的运行状态,末了是最主要的协程模块。

如何将SPP和Zend结合

SPP实在是基于协程的框架,协程是一个用户态的多线程观点。
在协程切换的时候会涉及内存管理的机制,而Zend没有这种切换内存资源的机制,只有全局变量和多线程资源隔离的办法。
这样的话要想将SPP和Zend结合起来,就要对Zend进行改造。

Zend的源码大概有60万行,如果直接改动核心源码,不只履行起来很麻烦,对之后的升级也会造成问题。
最好的办法是借助Zend本身的机制对入口进行改造,而不侵入内核。

Zend改造

Zend有多进程和多线程两种办法,在多线程模式下有一个线程安全的机制ZTS。
ZTS实质实在是对每个线程的全局资源进行了隔离,与SPP协程的结合就须要用到ZTS,下面是详细步骤。

第一步当然是打开Zend内核ZTS开关,第二步为了知足协程高下文切换,须要将ZTS中的线程私有变量转化为全局数据元素,第三步增加资源入口切换API。
做完这三步就完成了Zend和SPP的结合,虽然步骤不多但实际上在做的过程中还是会有很多寻衅。

PHP实行流调度器

办理了却合问题之后,接下来为了将全体流程串起来须要有一个实行流程调度器。
上图是全体实行流程,首先SPP通过SAPI进入到Zend中,然后Zend实行PHP脚本,先编译成OpCode,之后如果有网路IO就会用到协程。
协程也可以基于SPP供应的API来运作,通过Tsrm的全局资源table可以进行协程切换。

在有这样一套实行流的情形下,扩展也可以依赖SPP的API实现协程调度。

上图是SPP结合PHP之后的整体架构。
最底层是SPP,往上是PHP实行流调度器,中间是PHP内核和扩展,最上层是PHP脚本。
这一整套架构以嵌入式的办法结合了SPP和PHP,让开发者既可以利用SPP供应的高性能网络框架,同时也能运用到上层PHP快速开拓的特点。

这里展示的是压测时候的环境数据,包括机型、压测工具、压测方法以及框架的版本等。

这是压测后得到的实际数据比拟,可以看到SPP-PHP框架相对旧的框架性能上大概有3倍的提升。

以上为本日的分享内容,感激大家!

编者:IT大咖说,转载请标明版权和出处