开发一个简易的PHP扩展

本教程将实现一个阶乘扩展,编译环境为:CentOS7/PHP5.6.3,其PHP函数实现如下:

1
2
3
4
5
6
7
8
9
10
function factorial($number){
if($number < 1) return 0;
$i = 1;
$result = 1;
while($i <= $number){
$result *= $i;
$i++;
}
return $result;
}

注意:由于这个函数只用于演示,未考虑大数的阶乘,在使用大数进行阶乘时,有可能会发生整形溢出。

定义函数原型

我们首先在php源码的扩展目录定义一个文件名为”factorial.def”的文件,并在里面写上如下内容:

1
long factorial(int number)

生成扩展基本代码

这时候我们就可以使用php官方为我们提供的工具ext_skel,来生成扩展的基本代码,命令如下:

1
./ext_skel --extname=factorial --proto=factorial.def

其中extname表示要生成的扩展名称,proto表示函数原型的文件路径

扩展配置修改

经过上一步的代码生成,现在当前目录下应该出现了一个factorial的目录,我们进入这个目录,并编辑目录下的“config.m4”文件,将以下两行代码前的dnl删除,如下:

1
2
3
PHP_ARG_ENABLE(factorial, whether to enable factorial support,
Make sure that the comment is aligned:
[ --enable-factorial Enable factorial support])

功能实现

接下来就是我们今天的主菜了,我们修改一下factorial.c,找到函数主体,修改为如下代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
PHP_FUNCTION(factorial)
{
int argc = ZEND_NUM_ARGS();
long number;
if (zend_parse_parameters(argc TSRMLS_CC, "l", &number) == FAILURE)
return;
if(number < 1) RETURN_LONG(0);
long i = 1;
long result = 1;
while(i <= number){
result *= i;
i++;
}
RETURN_LONG(result);
}

代码中的zend_parse_parameters是用来检验输入参数的,它的第三个参数表示的是输入参数的类型,觉的类型表示如下表所示:

类型指定符 对应的C类型 描述
l long 长整数
d double 浮点数
s char *, int 二进制字符串,长度
b zend_bool 逻辑型(1或0)
r zval * 资源(文件指针,数据库连接等)
a zval * 联合数组
o zval * 任何类型的对象
O zval * 指定类型的对象。需要提供目标对象的类类型
z zval * 无任何操作的zval
foo foo foo
bar bar bar
baz baz baz

扩展函数的返回值,需要使用php预设的宏定义来返回,常见的宏定义如下表所示:

宏定义 设置返回值 宏返回类型和参数
RETURN_LONG(l) RETVAL_LONG(l) 整数
RETURN_BOOL(b) RETVAL_BOOL(b) 布尔数(1或0)
RETURN_NULL() RETVAL_NULL() NULL
RETURN_DOUBLE(d) RETVAL_DOUBLE(d) 浮点数
RETURN_STRING(s, dup) RETVAL_STRING(s, dup) 字符串。如果dup为1,引擎会调用estrdup()重复s,使用拷贝。如果dup为0,就使用s
RETURN_STRINGL(s, l, dup) RETVAL_STRINGL(s, l, dup) 长度为l的字符串值。与上一个宏一样,但因为s的长度被指定,所以速度更快。
RETURN_TRUE RETVAL_TRUE 返回布尔值true。注意到这个宏没有括号。
RETURN_FALSE RETVAL_FALSE 返回布尔值false。注意到这个宏没有括号。
RETURN_RESOURCE(r) RETVAL_RESOURCE(r) 资源句柄。

编译扩展

执行以下命令,执行扩展编译:

1
2
3
/usr/local/php/bin/phpzie
./configure
make && make install

添加扩展

编辑php.ini,加入如下代码:

1
extension=factorial.so

代码添加后,我们需要重启一下php-fpm.
至此,我们就将扩展添加进PHP了,这时我们可以编写一个php文件来做测试,如下:

1
2
echo factorial(10);
// returns 3628800