首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >让PHP能够调用C的函数-FFI扩展

让PHP能够调用C的函数-FFI扩展

作者头像
硬核项目经理
发布于 2020-06-23 08:12:41
发布于 2020-06-23 08:12:41
2.1K00
代码可运行
举报
运行总次数:0
代码可运行

让PHP能够调用C的函数-FFI扩展

在大型公司中,一般会有很多编程语言的配合。比如说让 Java 来做微服务层,用 C++ 来进行底层运算,用 PHP 来做中间层,最后使用 JS 展现效果。这些语言间的配合大部分都是通过 RPC 来完成,或者直接将数据入库再使用不同的语言来取用。那么,我们 PHP 的代码能否直接调用这些语言呢?其实,PHP 还真为我们准备了一个可以直接调用 C 语言的扩展库,并且这个扩展库还是已经默认内置在 PHP 中了,它就是 FFI 扩展。

什么是 FFI

FFI , Foreign Function Interface,外部函数接口。这个扩展允许我们加载一些公共库(.dll、.so),其实也就是可以调用一些 C 的数据结构及函数。它已经是随 PHP 源码发布的一个扩展了,在编译的时候可以加上 --with-ffi 来直接编译到 PHP 程序中。

我们这里已经是编译好的 PHP ,所以我们直接找到这个扩展,进行简单的扩展安装步骤就可以安装完成。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
cd php-7.4.4/ext/ffi/
phpize
./configure
make && make install

安装完成后记得在 php.ini 文件中打开扩展。关于这个扩展需要注意的一点是,它有一个配置项为 ffi.enable ,默认情况下这个配置项的值是 "preload" ,仅在 CLI SAPI 环境下启用 FFI 的能力。当然,我们也可以修改为 "true" 或 "false" 来开启和关闭它。设定为 "true" 将使得这个扩展在任何环境下都启用。

使用 FFI 调用 C 的函数

接下来,简单地看一下它是如何调用 C 的函数的。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 创建一个 FFI 对象,加载 libc 并且导入 printf 函数
$ffi_printf = FFI::cdef(
    "int printf(const char *format, ...);", // C 的定义规则
    "libc.so.6"); // 指定 libc 库
// 调用 C 的 printf 函数
$ffi_printf->printf("Hello %s!\n", "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 语言计算函数库。

定义变量和数组

当然,FFI 也是可以定义变量和数组的。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
// 创建一个 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\n", $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 语言的开发一样。

总结

是不是感觉很高大上?但是请注意哦,FFI 调用的 C 函数并没有 PHP 本身去调用的效率高。比如这种 pow() 函数,使用 PHP 自身的效率更好。而且,FFI 扩展虽说已经是跟随 PHP 同步发布的扩展,但它还是处于实验性质的。也就是说,这个扩展是为未来可能用到的其它功能准备的,而且还有很多不确定性。所以在生产环境中如果需要合适类似的功能的话,那么还是要做更多的深入调研哦。

测试代码:https://github.com/zhangyue0503/dev-blog/blob/master/php/202004/source/%E8%AE%A9PHP%E8%83%BD%E5%A4%9F%E8%B0%83%E7%94%A8C%E7%9A%84%E5%87%BD%E6%95%B0-FFI%E6%89%A9%E5%B1%95.php

参考文档:https://www.php.net/manual/zh/intro.ffi.phphttps://www.php.net/manual/zh/ffi.examples-basic.php

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2020-06-18,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 码农老张 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
今天你学C++了吗?——内存管理
前面在C语言阶段的一篇博客 C语言——动态内存分配我们提到过C/C++程序中内存区域的划分~
用户11352420
2024/12/25
1460
今天你学C++了吗?——内存管理
C&C++内存管理
在C语言中我们经常说,局部变量存放在栈区,动态内存开辟的空间是向堆区申请的,只读常量存放在常量区等等。其实这里我们所说的区域都是虚拟进程地址空间的一部分,具体划分如下:
始终学不会
2023/03/28
1.4K0
C&C++内存管理
C++:内存管理|new和delete
为什么需要内存管理呢??因为我们在程序的运行过程中会需要各种各样的数据,而我们根据数据的不同存储在不同的区域里面,是为了更高效地处理数据。而C语言相比Java来说在内存的权限上尽可能给了程序员更多的操作空间,这也是为什么C更追求性能。
小陈在拼命
2024/03/01
2510
C++:内存管理|new和delete
C++内存管理
方便管理,程序中需要不同类型的数据,主要是生命周期,比如当我们需要一个全局变量时,那么这个变量对应的是放在数据段中。还有一些局部变量,存放在栈中。
南桥
2024/07/26
1630
C++内存管理
【C++】动态内存管理
了解了这些之后,我们再来通过一个经典练习题深入理解一下内存区域的划分,如下代码:
修修修也
2024/04/20
2180
【C++】动态内存管理
C/C++内存管理
malloc和calloc的区别在于calloc会在返回地址之前把申请的空间的每个字节初始化为全0,malloc不为初始化,而realloc是扩容分为原地扩容和异地扩容。
用户11375356
2024/11/22
990
C/C++内存管理
C/C++内存管理
选项: A.栈 B.堆 C.数据段(静态区) D.代码段(常量区) globalVar在哪里?____ staticGlobalVar在哪里?____ staticVar在哪里?____ localVar在哪里?____ num1 在哪里?____ char2在哪里?____ *char2在哪里?____ pChar3在哪里?____ *pChar3在哪里?____ ptr1在哪里?____ *ptr1在哪里?____ 是不是有点乱,看一下图解吧。
小志biubiu
2025/02/27
1380
C/C++内存管理
[C++]内存管理
2.内存映射段是高效的I/O映射方式,用于装载一个共享的动态内存库。用户可使用系统接口创建共享共享内存,做进程间通信。
IT编程爱好者
2023/04/30
9860
[C++]内存管理
【C++内存管理】:new与delete,operator new与operator delete
📝前言: 上篇文章【C++高潮:类与对象】我们对C++的类与对象的知识点进行了讲解。 这篇文章我们在C语言内存管理的基础上探讨一下C++内存的管理: 1,C/C++内存分布 2,C语言内存管理 3,C++内存管理方式 4,operator new与operator delete 5,new和delete的实现原理 6,定位new表达式 7,malloc/free和new/delete的区别
用户11029137
2025/03/12
4640
【C++】内存管理
C++提供了对内存的精细控制,允许程序员以动态和手动的方式分配和释放内存。这种能力既带来强大的灵活性,也伴随着一定的挑战。
_小羊_
2024/10/16
1910
【C++】内存管理
C++从入门到精通——C++动态内存管理
C++动态内存管理涉及使用new和delete操作符来动态分配和释放堆内存。new用于在堆上分配内存并初始化对象,delete用于释放先前分配的内存。此外,C++还提供了智能指针如std::unique_ptr和std::shared_ptr来自动管理内存,以避免内存泄漏和悬挂指针。这些智能指针在超出作用域时会自动删除其所指向的对象。
鲜于言悠
2024/04/22
4360
C++从入门到精通——C++动态内存管理
【C++】探索C++内存管理:机制揭秘与内存安全
需要注意的是,C标准库中的malloc函数的具体实现可能因编译器和操作系统的不同而有所差异,上述步骤仅为一种常见的实现方式。
大耳朵土土垚
2024/05/03
2430
【C++】探索C++内存管理:机制揭秘与内存安全
【C/C++】图文题目吃透内存管理
学习目标:了解C/C++内存的分段情况,C++内容管理方式、operator new与operator delete函数 、new和delete的实现原理、定位new的表达式、最后介绍相关面试题的解析
平凡的人1
2022/11/15
1.2K0
【C/C++】图文题目吃透内存管理
初识C++ · 内存管理
语言不同,内存分布是相同的,对于局部变量都是放在栈上,全局变量都是放在静态区(数据段),动态开辟的都是从堆中开辟,const修饰的变量也是都放在常量区(代码段)
_lazy
2024/10/16
1440
初识C++ · 内存管理
【C++】C/C++内存管理
好的,并没有初始化。 那这样看的话,C++搞出new这些东西和C语言的malloc这些对于内置类型的操作好像除了用法之外也没有什么很大的区别。 那所以呢? C++搞出这些东西更多的是为了自定义类型,那new和delete操作自定义类型我们后面也会专门讲解,先不急。
YIN_尹
2024/01/23
2940
【C++】C/C++内存管理
C++第七弹 -- C/C++内存管理
在C/C++编程中,内存管理是至关重要的一个环节。程序员需要合理地分配和释放内存,以确保程序能够正常运行,避免内存泄漏和崩溃。本文将深入探讨C/C++内存管理机制,从内存分布、动态内存管理方式、new和delete的实现原理到定位new表达式,以及malloc/free和new/delete的区别,全面解析C/C++内存管理的方方面面。
用户11317877
2024/10/16
1900
C++第七弹 -- C/C++内存管理
C++奇迹之旅:C++内存管理的机制初篇
数据段:也叫静态数据段或初始化数据段,用于存储程序中的全局变量和静态变量,这些变量在程序启动时就已经分配好内存空间并初始化。 代码段:也叫文本段或指令段,用于存储程序的可执行指令代码。 这部分内存区域通常是只读的,程序在运行时不能修改代码段中的内容。
学习起来吧
2024/05/06
3190
C++奇迹之旅:C++内存管理的机制初篇
【C++】C/C++内存管理
程序的运行本质上就是存储一些指令,存储一些数据,对于数据,由于的需求的不同,有的可能使用一下就行了,有的需要长期使用,有的不能修改,因此内存中划分成不同的区域存放相关的一些数据(本文主要目的在于介绍C++相关内存管理方式,对于内存浅浅介绍一些知识,具体相关内存底层知识请移步其他文章。)
ZLRRLZ
2024/12/13
2350
【C++】C/C++内存管理
【C++】内存管理
1. 栈 又叫堆栈 -- 非静态局部变量 / 函数参数 / 返回值等等,栈是向下增长的。
啊QQQQQ
2024/11/19
1390
【C++】内存管理
【C++基础篇】学习C++就看这篇--->内存管理之new和delete
2️⃣内存映射段是高效的I/O映射方式,用于装载一个共享的动态内存库。用户可使用系统接口创建共享共享内存,做进程间通信。(在后续Linux博客的更新中会进行详细介绍)
HABuo
2025/07/15
1570
【C++基础篇】学习C++就看这篇--->内存管理之new和delete
相关推荐
今天你学C++了吗?——内存管理
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验