很多 PHPer 天天写表单,不知如何提升。如果你已经熟悉了数据集和服务器端的优化,不妨试试通过 PHP 扩展向系统层进军。当原有的 PHP 代码实现出现性能瓶颈,可以考虑通过 PHP 扩展实现;纯 PHP 代码无法实现的功能,可以考虑通过 PHP 扩展调用其他相关库实现。既能提升业务能力,又能帮助大家逐渐通过扩展了解 PHP 源代码层的运作机制。
扩展入门较难,这里特别推荐淘宝大牛信海龙老师的入门课程,手把手教会你只要一块钱。
我们精选出文集中的第二章内容,以供免费试读。如果你不喜欢付费课程,可以通过 PHP 手册第二部分进行学习(http://php.net/manual/zh/internals2.structure.php)
从hello world开始(试读)
以下内容以 PHP7 作为基础,讲解如何从零开始创建一个 PHP 扩展。示例中,我们将实现如下功能:
<?php
function say {
return "hello word";
}
echo say;
?>
输出内容:
$ php ./test.php
$ hello word
在扩展中实现一个 say 方法,调用 say 方法后,输出 hello word。
生成代码
PHP 为我们提供了生成基本代码的工具 ext_skel。这个工具在 PHP 源代码的 ./ext 目录下。
$ cd php_src/ext/
$ ./ext_skel --extname=say
extname 参数的值就是扩展名称。执行 ext_skel 命令后,这样在当前目录下会生成一个与扩展名一样的目录。
修改 config.m4 配置文件
config.m4 的作用就是配合 phpize 工具生成 configure 文件。configure 文件是用于环境检测的。检测扩展编译运行所需的环境是否满足。现在我们开始修改config.m4 文件。
$ cd ./say
$ vim ./config.m4
打开,config.m4 文件后,你会发现这样一段文字。
dnl If your extension references something external, use with:
dnl PHP_ARG_WITH(say, for say support,
dnl Make sure that the comment is aligned:
dnl [ --with-say Include say support])
dnl Otherwise use enable:
dnl PHP_ARG_ENABLE(say, whether to enable say support,
dnl Make sure that the comment is aligned:
dnl [ --enable-say Enable say support])
其中,dnl 是注释符号。
上面的代码说,如果你所编写的扩展如果依赖其它的扩展或者 lib 库,需要去掉 PHP_ARG_WITH 相关代码的注释。否则,去掉 PHP_ARG_ENABLE 相关代码段的注释。我们编写的扩展不需要依赖其他的扩展和 lib 库。
我们去掉 PHP_ARG_ENABLE 前面的注释。去掉注释后的代码如下:
dnl If your extension references something external, use with:
dnl PHP_ARG_WITH(say, for say support,
dnl Make sure that the comment is aligned:
dnl [ --with-say Include say support])
dnl Otherwise use enable:
PHP_ARG_ENABLE(say, whether to enable say support,
Make sure that the comment is aligned:
[ --enable-say Enable say support])
代码实现
修改 say.c 文件。实现 say 方法。 找到PHP_FUNCTION(confirm_say_compiled),在其上面增加如下代码:
PHP_FUNCTION(say)
{
zend_string *strg; strg = strpprintf(0, "hello word");
RETURN_STR(strg);
}
找到 PHP_FE(confirm_say_compiled, 在上面增加如下代码:
PHP_FE(say, )
修改后的代码如下:
const zend_function_entry say_functions = {
PHP_FE(say, ) /For testing, remove later. */
PHP_FE(confirm_say_compiled, ) /For testing, remove later. */
PHP_FE_ /Must be the last line in say_functions */
};
/}}} */
编译安装
编译扩展的步骤如下:
$ phpize
$ ./configure
$ make && make install
修改 php.ini 文件,增加如下代码:
[say]
extension = say.so
然后执行,php -m 命令。在输出的内容中,你会看到 say 字样。
调用测试
自己写一个脚本,调用 say 方法。看输出的内容是否符合预期。
<?php
echo say;
?>