您好,登錄后才能下訂單哦!
這篇文章主要介紹php中擴展錯誤如何返回,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!
php擴展返回錯誤的設置方法:1、打開相應的PHP文件;2、在擴展中通過“php_error_docref()”函數來拋出錯誤提示即可。
本文操作環境:windows7系統、PHP7.1版,DELL G3電腦
php 擴展錯誤怎么返回?
php源碼-擴展中拋出和處理錯誤
先說說源碼層面的錯誤種類,大概有下面幾種
//zend_errors.h 文件 #define E_ERROR (1<<0L) #define E_WARNING (1<<1L) #define E_PARSE (1<<2L) #define E_NOTICE (1<<3L) #define E_CORE_ERROR (1<<4L) #define E_CORE_WARNING (1<<5L) #define E_COMPILE_ERROR (1<<6L) #define E_COMPILE_WARNING (1<<7L) #define E_USER_ERROR (1<<8L) #define E_USER_WARNING (1<<9L) #define E_USER_NOTICE (1<<10L) #define E_STRICT (1<<11L) #define E_RECOVERABLE_ERROR (1<<12L) #define E_DEPRECATED (1<<13L) #define E_USER_DEPRECATED (1<<14L)
其中 E_CORE_ERROR, E_ERROR, E_RECOVERABLE_ERROR, E_PARSE, E_COMPILE_ERROR,E_USER_ERROR, 這種錯誤會觸發try catch 異常處理流程,也就是會中斷當前request的執行,發生這種錯誤時會把當前要執行的opcode設置為 ZEND_HANDLE_EXCEPTION, 從而跳出程序執行 ,具體跳出過程參考: php源碼-異常throw處理過程-02
我們在寫擴展的時候如何拋出錯誤提示呢?
擴展中可以通過 php_error_docref()函數來拋出錯誤提示,比如
PHP_FUNCTION(academy_sample_fopen) { FILE *fp; char *filename, *mode; int filename_len, mode_len; if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "ss", &filename, &filename_len, &mode, &mode_len) == FAILURE) { RETURN_NULL(); } if (!filename_len || !mode_len) { php_error_docref(NULL TSRMLS_CC, E_WARNING,"Invalid filename or mode length"); RETURN_FALSE; } fp = fopen(filename, mode); if (!fp) { php_error_docref(NULL TSRMLS_CC, E_WARNING,"Unable to open %s using mode %s", filename, mode); RETURN_FALSE; } }
通過php_error_docref這是的錯誤提示,如果觸發的時候會出現類型下面的錯誤提示輸出
相信在寫php代碼的時候也都見過這樣類似的錯誤提示輸出
PHP Fatal error: Unknown: EEEEEEEEEEEEEEEEEEE in Unknown on line 0 PHP Warning: Swoole\Php\Runner::run() expects exactly 4 parameters, 0 given in /var/www/swoole/http_test.php on line 22
這個錯誤提示輸出是如何實現的呢?
跟進php_error_docref的源碼
//main/php.h #define php_error_docref php_error_docref0 //main/main.c PHPAPI ZEND_COLD void php_error_docref0(const char *docref, int type, const char *format, ...) { va_list args; va_start(args, format); php_verror(docref, "", type, format, args); va_end(args); } //main/main.c PHPAPI ZEND_COLD void php_verror(const char *docref, const char *params, int type, const char *format, va_list args) { php_error(type, "%s", message); efree(message); } //main/php.h #define php_error zend_error //Zend/zend.c ZEND_API ZEND_COLD void zend_error(int type, const char *format, ...) /* {{{ */ { va_list va; va_start(va, format); zend_error_va_list(type, format, va); va_end(va); } static ZEND_COLD void zend_error_va_list(int type, const char *format, va_list args) { if (EG(exception)) { switch (type) { case E_CORE_ERROR: case E_ERROR: case E_RECOVERABLE_ERROR: case E_PARSE: case E_COMPILE_ERROR: case E_USER_ERROR: //嚴重錯誤,通過ZEND_HANDLE_EXCEPTION 中斷程序 if (ex && ex->opline->opcode == ZEND_HANDLE_EXCEPTION && EG(opline_before_exception)) { opline = EG(opline_before_exception); } break; } } //... // zend_error_cb 很重要, 這個函數是在 sapi啟動的時候,通過 php_module_startup() 賦值為 php_error_cb() 函數 zend_error_cb(type, error_filename, error_lineno, format, args); }
zend_error_cb 很重要, 這個函數是在 sapi啟動的時候,通過 php_module_startup() 賦值為 php_error_cb()函數 , 而php_error_cb() 最終會調用 _sapi_module_struct.log_message(), 也就是當你調用php_error_docref()函數來拋出錯誤的時候,實際上會回調 _sapi_module_struct.log_message (具體執行過程參考:php源碼-sapi中自定義錯誤輸出), 比如fpm sapi就會把錯誤信息返回給, cli sapi通過自定義這個函數把錯誤信息輸出到標準輸出
以上是“php中擴展錯誤如何返回”這篇文章的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業資訊頻道!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。