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 可以 点击这儿
写个简单的扩展举例,通过php扩展的方式来实现python中的all
和 any
函数。
准备工作。
- 下载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.phpt
和003.phpt
, 新建 004.phpt
和005.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新增了蛮多有用的宏和特性。
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。