http://blog.csdn.net/super_ufo/article/details/3863731 php
php調用C代碼的方法詳解 html
在php程序中須要用到C代碼,應該是下面兩種狀況: node
Boolean | b | zend_bool |
Long | l | long |
Double | d | double |
String | s | char*, int |
Resource | r | zval* |
Array | a | zval* |
Object | o | zval* |
zval | z | zval* |
Traditional | Non-Persistent | Persistent |
---|---|---|
malloc(count) calloc(count, num) |
emalloc(count) ecalloc(count, num) |
pemalloc(count, 1)* pecalloc(count, num, 1) |
strdup(str) strndup(str, len) |
estrdup(str) estrndup(str, len) |
pestrdup(str, 1) pemalloc() & memcpy() |
free(ptr) | efree(ptr) | pefree(ptr, 1) |
realloc(ptr, newsize) | erealloc(ptr, newsize) | perealloc(ptr, newsize, 1) |
malloc(count * num + extr)** | safe_emalloc(count, num, extr) | safe_pemalloc(count, num, extr) |
更好的文章:http://www.toplee.com/blog/56.html#pp1
程序員
本節沒有介紹關於腳本引擎基本構造的一些知識,而是直接進入擴展的編碼講解中,所以不要擔憂你沒法馬上得到對擴展總體把握的感受。假設你正在開發一個網站,須要一個把字符串重複n次的函數。下面是用PHP寫的例子: 算法
function self_concat($string, $n) 數據庫
{ vim
$result = ""; windows
for ($i = 0; $i < $n; $i++) { 數組
$result .= $string; 安全
}
return $result;
}
self_concat("One", 3) returns "OneOneOne".
self_concat("One", 1) returns "One".
假設因爲一些奇怪的緣由,你須要時常調用這個函數,並且還要傳給函數很長的字符串和大值n。這意味着在腳本里有至關巨大的字符串鏈接量和內存從新分配過程,以致顯著地下降腳本執行速度。若是有一個函數可以更快地分配大量且足夠的內存來存放結果字符串,而後把$string重複n次,就不須要在每次循環迭代中分配內存。
爲擴展創建函數的第一步是寫一個函數定義文件,該函數定義文件定義了擴展對外提供的函數原形。該例中,定義函數只有一行函數原形self_concat() :
string self_concat(string str, int n)
函數定義文件的通常格式是一個函數一行。你能夠定義可選參數和使用大量的PHP類型,包括: bool, float, int, array等。
保存爲myfunctions.def文件至PHP原代碼目錄樹下。
該是經過擴展骨架(skeleton)構造器運行函數定義文件的時機了。該構造器腳本叫ext_skel,放在PHP原代碼目錄樹的ext/目錄下(PHP原碼主目錄下的README.EXT_SKEL提供了更多的信息)。假設你把函數定義保存在一個叫作myfunctions.def的文件裏,並且你但願把擴展取名爲myfunctions,運行下面的命令來創建擴展骨架
./ext_skel --extname=myfunctions --proto=myfunctions.def
這個命令在ext/目錄下創建了一個myfunctions/目錄。你要作的第一件事情也許就是編譯該骨架,以便編寫和測試實際的C代碼。編譯擴展有兩種方法:
☞ 做爲一個可裝載模塊或者DSO(動態共享對象)
☞ 靜態編譯到PHP
由於第二種方法比較容易上手,因此本章採用靜態編譯。若是你對編譯可裝載擴展模塊感興趣,能夠閱讀PHP原代碼根目錄下的README.SELF-CONTAINED_EXTENSIONS文件。爲了使擴展可以被編譯,須要修改擴展目錄ext/myfunctions/下的config.m4文件。擴展沒有包裹任何外部的C庫,你須要添加支持--enable-myfunctions配置開關到PHP編譯系統裏(–with-extension 開關用於那些須要用戶指定相關C庫路徑的擴展)。能夠去掉自動生成的下面兩行的註釋來開啓這個配置。
PHP_ARG_ENABLE(myfunctions, whether to enable myfunctions support,
[ --enable-myfunctions Include myfunctions support])
如今剩下的事情就是在PHP原代碼樹根目錄下運行./buildconf,該命令會生成一個新的配置腳本。經過查看./configure --help輸出信息,能夠檢查新的配置選項是否被包含到配置文件中。如今,打開你喜愛的配置選項開關和--enable-myfunctions從新配置一下PHP。最後的但不是最次要的是,用make來從新編譯PHP。
ext_skel應該把兩個PHP函數添加到你的擴展骨架了:打算實現的self_concat()函數和用於檢測myfunctions 是否編譯到PHP的confirm_myfunctions_compiled()函數。完成PHP的擴展開發後,能夠把後者去掉。
<?php
print confirm_myfunctions_compiled("myextension");
?>
運行這個腳本會出現相似下面的輸出:
"Congratulations! You have successfully modified ext/myfunctions
config.m4. Module myfunctions is now compiled into PHP."
另外,ext_skel腳本生成一個叫myfunctions.php的腳本,你也能夠利用它來驗證擴展是否被成功地編譯到PHP。它會列出該擴展所支持的全部函數。
如今你學會如何編譯擴展了,該是真正地研究self_concat()函數的時候了。
下面就是ext_skel腳本生成的骨架結構:
/* {{{ proto string self_concat(string str, int n)
*/
PHP_FUNCTION(self_concat)
}
char *str = NULL;
int argc = ZEND_NUM_ARGS();
int str_len;
long n;
if (zend_parse_parameters(argc TSRMLS_CC, "sl", &str, &str_len, &n) == FAILURE)
return;
php_error(E_WARNING, "self_concat: not yet implemented");
}
/* }}} */
自動生成的PHP函數週圍包含了一些註釋,這些註釋用於自動生成代碼文檔和vi、Emacs等編輯器的代碼摺疊。函數自身的定義使用了宏PHP_FUNCTION(),該宏能夠生成一個適合於Zend引擎的函數原型。邏輯自己分紅語義各部分,取得調用函數的參數和邏輯自己。
爲了得到函數傳遞的參數,可使用zend_parse_parameters()API函數。下面是該函數的原型:
zend_parse_parameters(int num_args TSRMLS_DC, char *type_spec, …);
第一個參數是傳遞給函數的參數個數。一般的作法是傳給它ZEND_NUM_ARGS()。(ZEND_NUM_ARGS() 來表示對傳入的參數「有多少要多少」)這是一個表示傳遞給函數參數總個數的宏。第二個參數是爲了線程安全,老是傳遞TSRMLS_CC宏,後面會講到。第三個參數是一個字符串,指定了函數指望的參數類型,後面緊跟着須要隨參數值更新的變量列表。由於PHP採用鬆散的變量定義和動態的類型判斷,這樣作就使得把不一樣類型的參數轉化爲指望的類型成爲可能。例如,若是用戶傳遞一個整數變量,可函數須要一個浮點數,那麼zend_parse_parameters()就會自動地把整數轉換爲相應的浮點數。若是實際值沒法轉換成指望類型(好比整形到數組形),會觸發一個警告。
下表列出了可能指定的類型。咱們從完整性考慮也列出了一些沒有討論到的類型。
類型指定符 |
對應的C類型 |
描述 |
l |
long |
符號整數 |
d |
double |
浮點數 |
s |
char *, int |
二進制字符串,長度 |
b |
zend_bool |
邏輯型(1或0) |
r |
zval * |
資源(文件指針,數據庫鏈接等) |
a |
zval * |
聯合數組 |
o |
zval * |
任何類型的對象 |
O |
zval * |
指定類型的對象。須要提供目標對象的類類型 |
z |
zval * |
無任何操做的zval |
爲了容易地理解最後幾個選項的含義,你須要知道zval是Zend引擎的值容器[1]。不管這個變量是布爾型,字符串型或者其餘任何類型,其信息總會包含在一個zval聯合體中。本章中咱們不直接存取zval,而是經過一些附加的宏來操做。下面的是或多或少在C中的zval, 以便咱們能更好地理解接下來的代碼。
typedef union _zval {
long lval;
double dval;
struct {
char *val;
int len;
} str;
HashTable *ht;
zend_object_value obj;
} zval;
在咱們的例子中,咱們用基本類型調用zend_parse_parameters(),以本地C類型的方式取得函數參數的值,而不是用zval容器。
爲了讓zend_parse_parameters()可以改變傳遞給它的參數的值,並返回這個改變值,須要傳遞一個引用。仔細查看一下self_concat():
if (zend_parse_parameters(argc TSRMLS_CC, "sl", &str, &str_len, &n) == FAILURE)
return;
注意到自動生成的代碼會檢測函數的返回值FAILUER(成功即SUCCESS)來判斷是否成功。若是沒有成功則當即返回,而且由zend_parse_parameters()負責觸發警告信息。由於函數打算接收一個字符串l和一個整數n,因此指定 」sl」 做爲其類型指示符。s須要兩個參數,因此咱們傳遞參考char * 和 int (str 和 str_len)給zend_parse_parameters()函數。不管何時,記得老是在代碼中使用字符串長度str_len來確保函數工做在二進制安全的環境中。不要使用strlen()和strcpy(),除非你不介意函數在二進制字符串下不能工做。二進制字符串是包含有nulls的字符串。二進制格式包括圖象文件,壓縮文件,可執行文件和更多的其餘文件。」l」 只須要一個參數,因此咱們傳遞給它n的引用。儘管爲了清晰起見,骨架腳本生成的C變量名與在函數原型定義文件中的參數名同樣;這樣作不是必須的,儘管在實踐中鼓勵這樣作。
回到轉換規則中來。下面三個對self_concat()函數的調用使str, str_len和n獲得一樣的值:
self_concat("321", 5);
self_concat(321, "5");
self_concat("321", "5");
str points to the string "321", str_len equals 3, and n equals 5.
str 指向字符串"321",str_len等於3,n等於5。
在咱們編寫代碼來實現鏈接字符串返回給PHP的函數前,還得談談兩個重要的話題:內存管理、從PHP內部返回函數值所使用的API!!
用於從堆中分配內存的PHP API幾乎和標準C API同樣。在編寫擴展的時候,使用下面與C對應(所以沒必要再解釋)的API函數:
emalloc(size_t size);
efree(void *ptr);
ecalloc(size_t nmemb, size_t size);
erealloc(void *ptr, size_t size);
estrdup(const char *s);
estrndup(const char *s, unsigned int length);
在這一點上,任何一位有經驗的C程序員應該象這樣思考一下:「什麼?標準C沒有strndup()?」是的,這是正確的,由於GNU擴展一般在Linux下可用。estrndup()只是PHP下的一個特殊函數。它的行爲與estrdup()類似,可是能夠指定字符串重複的次數(不須要結束空字符),同時是二進制安全的。這是推薦使用estrndup()而不是estrdup()的緣由。
在幾乎全部的狀況下,你應該使用這些內存分配函數。有一些狀況,即擴展須要分配在請求中永久存在的內存,從而不得不使用malloc(),可是除非你知道你在作什麼,你應該始終使用以上的函數。若是沒有使用這些內存函數,而相反使用標準C函數分配的內存返回給腳本引擎,那麼PHP會崩潰。
這些函數的優勢是:任何分配的內存在偶然狀況下若是沒有被釋放,則會在頁面請求的最後被釋放。所以,真正的內存泄漏不會產生。然而,不要依賴這一機制,從調試和性能兩個緣由來考慮,應當確保釋放應該釋放的內存。剩下的優勢是在多線程環境下性能的提升,調試模式下檢測內存錯誤等。
還有一個重要的緣由,你不須要檢查這些內存分配函數的返回值是否爲null。當內存分配失敗,它們會發出E_ERROR錯誤,從而決不會返回到擴展。
擴展API包含豐富的用於從函數中返回值的宏。這些宏有兩種主要風格:第一種是RETVAL_type()形式,它設置了返回值但C代碼繼續執行。這一般使用在把控制交給腳本引擎前還但願作的一些清理工做的時候使用,而後再使用C的返回聲明 」return」 返回到PHP;後一個宏更加廣泛,其形式是RETURN_type(),他設置了返回類型,同時返回控制到PHP。下表解釋了大多數存在的宏。
設置返回值而且結束函數 |
設置返回值 |
宏返回類型和參數 |
RETURN_LONG(l) |
RETVAL_LONG(l) |
整數 |
RETURN_BOOL(b) |
RETVAL_BOOL(b) |
布爾數(1或0) |
RETURN_NULL() |
RETVAL_NULL() |
NULL |
RETURN_DOUBLE(d) |
RETVAL_DOUBLE(d) |
浮點數 |
RETURN_STRING(s, dup) |
RETVAL_STRING(s, dup) |
字符串。若是dup爲1,引擎會調用estrdup()重複s,使用拷貝。若是dup爲0,就使用s |
RETURN_STRINGL(s, l, dup) |
RETVAL_STRINGL(s, l, dup) |
長度爲l的字符串值。與上一個宏同樣,但由於s的長度被指定,因此速度更快。 |
RETURN_TRUE |
RETVAL_TRUE |
返回布爾值true。注意到這個宏沒有括號。 |
RETURN_FALSE |
RETVAL_FALSE |
返回布爾值false。注意到這個宏沒有括號。 |
RETURN_RESOURCE(r) |
RETVAL_RESOURCE(r) |
資源句柄。 |