傳回值
使用者空間函數利用return關鍵字向它的呼叫空間回傳訊息, 這一點和C語言的語法相同.
例如:
function sample_long() { return 42; } $bar = sample_long();
當sample_long()被呼叫時, 返回42並且設定到$bar變數中. 在C語言中的等價程式碼如下:
int sample_long(void) { return 42; } void main(void) { int bar = sample_long(); }
當然,在C語言中你總是知道被調用的函數是什麼, 並且基於函數原型返回, 因此相應的你要定義返回結果存儲的變量. 在php用戶空間處理時, 變量類型是動態的, 轉而依賴於在第2章"變數的里里外外"中介紹的zval的類型.
return_value變數
你可能認為你的內部函數應該直接回傳一個zval , 或者說分配一個zval的記憶體空間並如下返回zval *.
PHP_FUNCTION(sample_long_wrong) { zval *retval; MAKE_STD_ZVAL(retval); ZVAL_LONG(retval, 42); return retval; }
不幸的是, 這樣做是不正確的. 並不是強制每個函數實現分配zval並返回它. 而是Zend引擎在函數呼叫之前預先分配這個空間. 接著將zval的類型初始化為IS_NULL, 並將值作為參數名return_value傳遞. 以下是正確的做法:
PHP_FUNCTION(sample_long) { ZVAL_LONG(return_value, 42); return; }
要注意的是PHP_FUNCTION()實現並不會直接回傳任何值. 取而代之的是直接將適當的資料彈出到return_value參數中, Zend引擎會在內部函數執行完成後處理它.
友情提示: ZVAL_LONG()宏是對多個賦值操作的一個封裝:
Z_TYPE_P(return_value) = IS_LONG; Z_LVAL_P(return_value) = 42;
或更直接點:
return_value->type = IS_LONG; return_value->value.lval = 42;
return_value的is_ref和refcount屬性不應該被內部函數直接修改. 這些值由Zend引擎在呼叫你的函數時初始化並處理.
現在我們來看看這個特殊的函數, 將它增加到第5章"你的第一個擴展"中創建的sample擴展中. 只需要在sample_hello_world()函數下面加入這個函式, 並將sample_long()加入到php_sample_functions結構體中:
static function_entry php_sample_functions[] = { PHP_FE(sample_hello_world, NULL) PHP_FE(sample_long, NULL) { NULL, NULL, NULL } };
現在我們就可以執行make重新建構擴充了.
如果一切OK, 可以運行php並測試新函數:
$ php -r 'var_dump(sample_long());
包裝更緊湊的巨集
在程式碼可讀性和可維護性方面, ZVAL_*()宏中有重複的部分: return_value變數. 這種情況下, 將巨集的ZVAL替換為RETVAL, 我們就可以在呼叫時省略return_value了.
前面的例子中, sample_long()的實作程式碼可以縮減到下面這樣:
PHP_FUNCTION(sample_long) { RETVAL_LONG(42); return; }
以上是php自訂函數傳回值的用法詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!