
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!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 語言計算函數庫。
定義變量和數組
// 創建一個 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 : %fn", $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視頻教程》
站長資訊網