如何在PHP中调用C的函数
FFI
提供了高级语言直接的互相调用,而对于PHP
来说,FFI
让我们可以方便的调用C
语言写的各种库。其实现有大量的PHP扩展是对一些已有的C库的包装,比如常用的mysqli
, curl
, gettext
等,PECL
中也有大量的类似扩展。传统的方式,当我们需要用一些已有的C
语言的库的能力的时候,我们需要用C语言写wrapper
,把他们包装成扩展,这个过程中就需要大家去学习PHP
的扩展怎么写,当然现在也有一些方便的方式,比如Zephir.
但总还是有一些学习成本的,而有了FFI
以后,我们就可以直接在PHP
脚本中调用C
语言写的库中的函数了。而C
语言几十年的历史中,积累了大量的优秀的库,FFI
直接让我们可以方便的享受这个庞大的资源了。
注意:FFI
应在PHP7.4
版本后使用
// 创建一个 FFI 对象,加载 libc 并且导入 printf 函数 $ffi_printf = FFI::cdef( "int printf(const char *format, ...);", // C 的定义规则 "libc.so.6"); // 指定 libc 库 // 调用 C 的 printf 函数 $ffi_printf->printf("Hello %s! ", "world"); // Hello World // 加载 math 并且导入 pow 函数 $ffi_pow = FFI::cdef( "double pow(double x, double y);", "libboost_math_c99.so.1.66.0"); // 这里调用的是 C 的 pow 函数,不是 PHP 自己的 echo $ffi_pow->pow(2,3), PHP_EOL; // 8
创建了两个对象,分别调用了 C
的 printf()
和 pow()
函数。FFI::cdef()
是用于创建一个 FFI
对象,它接收两个参数,一个是包含常规C语言
(类型、结构、函数、变量等)声明序列的字符串。实际上,这个字符串可以从C头文件复制粘贴。而另一个参数则是要加载并定义链接的共享库文件的名称。也就是我们需要的 .dll
或 .so
文件,它与我们声明字符串是对应的,比如在 libc.so.6
中并没有 pow()
这类的计算函数,所以我们就要找到 math
相关的 C
语言计算函数库。
定义变量和数组
// 创建一个 int 变量 $x = FFI::new("int"); var_dump($x->cdata); // int(0) // 为变量赋值 $x->cdata = 5; var_dump($x->cdata); // int(5) // 计算变量 $x->cdata += 2; var_dump($x->cdata); // int(7) // 结合上面的两个 FFI 对象操作 echo "pow value:", $ffi_pow->pow($x->cdata, 3), PHP_EOL; // pow value:343 $ffi_printf->printf("Int Pow value is : %f ", $ffi_pow->pow($x->cdata, 3)); // Int Pow value is : 343.000000 // 创建一个数组 $a = FFI::new("long[1024]"); // 为数组赋值 for ($i = 0; $i < count($a); $i++) { $a[$i] = $i; } var_dump($a[25]); // int(25) $sum = 0; foreach ($a as $n) { $sum += $n; } var_dump($sum); // int(523776) var_dump(count($a)); // int(1024) 数组长度 var_dump(FFI::sizeof($a)); // int(8192),内存大小
使用 FFI::new()
函数来创建一个 C
的数据结构,也就是变量声明,这些变量的内容将保存在 cdata
属性中。而数组则直接就可以操作这个函数的返回值。当然,当我们要结束使用的时候,还是需要使用 FFI::free()
来释放变量的,就和 C
语言的开发一样。
推荐:《2021年PHP面试题大汇总(收藏)》《php视频教程》