溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

如何在PHP中調(diào)用C函數(shù)

發(fā)布時(shí)間:2021-06-02 16:35:08 來(lái)源:億速云 閱讀:145 作者:Leah 欄目:編程語(yǔ)言

如何在PHP中調(diào)用C函數(shù)?針對(duì)這個(gè)問題,這篇文章詳細(xì)介紹了相對(duì)應(yīng)的分析和解答,希望可以幫助更多想解決這個(gè)問題的小伙伴找到更簡(jiǎn)單易行的方法。

什么是 FFI

FFI , Foreign Function Interface,外部函數(shù)接口。這個(gè)擴(kuò)展允許我們加載一些公共庫(kù)(.dll、.so),其實(shí)也就是可以調(diào)用一些 C 的數(shù)據(jù)結(jié)構(gòu)及函數(shù)。它已經(jīng)是隨 PHP 源碼發(fā)布的一個(gè)擴(kuò)展了,在編譯的時(shí)候可以加上 --with-ffi 來(lái)直接編譯到 PHP 程序中。

我們這里已經(jīng)是編譯好的 PHP ,所以我們直接找到這個(gè)擴(kuò)展,進(jìn)行簡(jiǎn)單的擴(kuò)展安裝步驟就可以安裝完成。

cd php-7.4.4/ext/ffi/
phpize
./configure
make && make install

安裝完成后記得在 php.ini 文件中打開擴(kuò)展。關(guān)于這個(gè)擴(kuò)展需要注意的一點(diǎn)是,它有一個(gè)配置項(xiàng)為 ffi.enable ,默認(rèn)情況下這個(gè)配置項(xiàng)的值是 "preload" ,僅在 CLI SAPI 環(huán)境下啟用 FFI 的能力。當(dāng)然,我們也可以修改為 "true" 或 "false" 來(lái)開啟和關(guān)閉它。設(shè)定為 "true" 將使得這個(gè)擴(kuò)展在任何環(huán)境下都啟用。

使用 FFI 調(diào)用 C 的函數(shù)

接下來(lái),簡(jiǎn)單地看一下它是如何調(diào)用 C 的函數(shù)的。

// 創(chuàng)建一個(gè) FFI 對(duì)象,加載 libc 并且導(dǎo)入 printf 函數(shù)
$ffi_printf = FFI::cdef(
    "int printf(const char *format, ...);", // C 的定義規(guī)則
    "libc.so.6"); // 指定 libc 庫(kù)
// 調(diào)用 C 的 printf 函數(shù)
$ffi_printf->printf("Hello %s!\n", "world"); // Hello World

// 加載 math 并且導(dǎo)入 pow 函數(shù)
$ffi_pow = FFI::cdef(
    "double pow(double x, double y);", 
    "libboost_math_c99.so.1.66.0");
// 這里調(diào)用的是 C 的 pow 函數(shù),不是 PHP 自己的
echo $ffi_pow->pow(2,3), PHP_EOL; // 8

我們創(chuàng)建了兩個(gè)對(duì)象,分別調(diào)用了 C 的 printf() 和 pow() 函數(shù)。FFI::cdef() 是用于創(chuàng)建一個(gè) FFI 對(duì)象,它接收兩個(gè)參數(shù),一個(gè)是包含常規(guī)C語(yǔ)言(類型、結(jié)構(gòu)、函數(shù)、變量等)聲明序列的字符串。實(shí)際上,這個(gè)字符串可以從C頭文件復(fù)制粘貼。而另一個(gè)參數(shù)則是要加載并定義鏈接的共享庫(kù)文件的名稱。也就是我們需要的 .dll 或 .so 文件,它與我們聲明字符串是對(duì)應(yīng)的,比如在 libc.so.6 中并沒有 pow() 這類的計(jì)算函數(shù),所以我們就要找到 math 相關(guān)的 C 語(yǔ)言計(jì)算函數(shù)庫(kù)。

定義變量和數(shù)組

當(dāng)然,F(xiàn)FI 也是可以定義變量和數(shù)組的。

// 創(chuàng)建一個(gè) int 變量
$x = FFI::new("int");
var_dump($x->cdata); // int(0)

// 為變量賦值
$x->cdata = 5;
var_dump($x->cdata); // int(5)

// 計(jì)算變量
$x->cdata += 2;
var_dump($x->cdata); // int(7)


// 結(jié)合上面的兩個(gè) FFI 對(duì)象操作

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


// 創(chuàng)建一個(gè)數(shù)組
$a = FFI::new("long[1024]");
// 為數(shù)組賦值
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) 數(shù)組長(zhǎng)度
var_dump(FFI::sizeof($a)); // int(8192),內(nèi)存大小

使用 FFI::new() 函數(shù)來(lái)創(chuàng)建一個(gè) C 的數(shù)據(jù)結(jié)構(gòu),也就是變量聲明,這些變量的內(nèi)容將保存在 cdata 屬性中。而數(shù)組則直接就可以操作這個(gè)函數(shù)的返回值。當(dāng)然,當(dāng)我們要結(jié)束使用的時(shí)候,還是需要使用 FFI::free() 來(lái)釋放變量的,就和 C 語(yǔ)言的開發(fā)一樣。

關(guān)于如何在PHP中調(diào)用C函數(shù)問題的解答就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關(guān)注億速云行業(yè)資訊頻道了解更多相關(guān)知識(shí)。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

php
AI