首頁  >  文章  >  後端開發  >  php自訂函數傳回值的用法詳解

php自訂函數傳回值的用法詳解

伊谢尔伦
伊谢尔伦原創
2017-06-27 09:15:116380瀏覽

傳回值

使用者空間函數利用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中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn