php8提供了非常方便的扩展函数或类参数信息的生成工具

只需要维护一份xyz.stub.php,就可以使用工具生成 xyz_arginfo.h

毫无疑问,这种方式,又降低了广大 phper 开发扩展的门槛,更易维护。

上手体验:

生成扩展骨架。

cd ext
php ext_skel.php --ext test

随便添加一个函数,更改 test.stub.php

<?php

/** @generate-function-entries */

function test1(): void {}

function test2(string $str = ""): string {}

function test3(int $integer = 123): int {}

重新生成 test_arginfo.h

php ../../build/gen_stub.php test.stub.php

相关 commit 可以 点击这儿(https://github.com/php/php-src/compare/master…nikic:php-stubs)

写个简单的扩展举例,通过php扩展的方式来实现python中的allany函数。

准备工作。下载php最新源码已经安装好php生成扩展骨架。

cd ext
php ext_skel.php --ext python

撰写函数原型,编辑 python.stub.php

<?php

/** @generate-function-entries */

function all(array $arr): bool {}

function any(array $arr): bool {}

根据 python.stub.php 生成 python_arginfo.h

php ../../build/gen_stub.php python.stub.php

实现函数逻辑,编辑 python.c

PHP_FUNCTION(all)
{
    zval *input;
    zval *item;
    int result = 1, item_result = 1;
    HashTable *htbl;

    ZEND_PARSE_PARAMETERS_START(1, 1)
        Z_PARAM_ARRAY(input)
    ZEND_PARSE_PARAMETERS_END();

    htbl = Z_ARRVAL_P(input);

    ZEND_HASH_FOREACH_VAL(htbl, item) {
        item_result = zend_is_true(item);
        result &= item_result;
    } ZEND_HASH_FOREACH_END();

    RETURN_BOOL(result);
}

/* {{{ void any() */
PHP_FUNCTION(any)
{
    zval *input;
    zval *item;
    int result = 0, item_result = 0;
    HashTable *htbl;

    ZEND_PARSE_PARAMETERS_START(1, 1)
        Z_PARAM_ARRAY(input)
    ZEND_PARSE_PARAMETERS_END();

    htbl = Z_ARRVAL_P(input);

    ZEND_HASH_FOREACH_VAL(htbl, item) {
        item_result = zend_is_true(item);
        result |= item_result;
    } ZEND_HASH_FOREACH_END();

    RETURN_BOOL(result);
}

编写单元测试,编辑 002.phpt003.phpt, 新建 004.phpt005.phpt

--TEST--
Check all function true case
--SKIPIF--
<?php
if (!extension_loaded('python')) {
    echo 'skip';
}
?>
--FILE--
<?php
var_dump(all([]));
var_dump(all([1]));
var_dump(all([-1, 1, '1']));
?>
--EXPECT--
bool(true)
bool(true)
bool(true)
--TEST--
Check all function false case
--SKIPIF--
<?php
if (!extension_loaded('python')) {
    echo 'skip';
}
?>
--FILE--
<?php
var_dump(all(['0']));
var_dump(all([0]));
var_dump(all(['']));
var_dump(all([false]));
var_dump(all([1, -1, 100, false]));
var_dump(all([0, -1, 100, 1]));
var_dump(all(['1', -1, '', 100, 1]));
?>
--EXPECT--
bool(false)
bool(false)
bool(false)
bool(false)
bool(false)
bool(false)
bool(false)
--TEST--
Check any function true case
--SKIPIF--
<?php
if (!extension_loaded('python')) {
    echo 'skip';
}
?>
--FILE--
<?php
var_dump(any(['a', 'b', 'c', 'd']));
var_dump(any([['a', 'b', '', 'd']]));
var_dump(any([['', 'b', 'c', 'd']]));
var_dump(any([['a', 'b', 'c', '']]));
?>
--EXPECT--
bool(true)
bool(true)
bool(true)
bool(true)
--TEST--
Check all function false case
--SKIPIF--
<?php
if (!extension_loaded('python')) {
    echo 'skip';
}
?>
--FILE--
<?php
var_dump(any(['0']));
var_dump(any([0]));
var_dump(any(['']));
var_dump(any([false]));
var_dump(any([0, '0', '', false]));
?>
--EXPECT--
bool(false)
bool(false)
bool(false)
bool(false)
bool(false)

编译、测试和安装

./configure && make
make test
sudo make install

加入到php.ini

php -i | grep ini # 定位你的php.ini文件

加入

extension=python.so

查看是否成功

php -m | grep python

实测

php -r "var_dump(all([]));“
php -r "var_dump(any([]));"

PHP8新增了蛮多有用的宏和特性。