溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

php中擴(kuò)展錯(cuò)誤如何返回

發(fā)布時(shí)間:2021-09-14 10:04:14 來(lái)源:億速云 閱讀:144 作者:小新 欄目:編程語(yǔ)言

這篇文章主要介紹php中擴(kuò)展錯(cuò)誤如何返回,文中介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們一定要看完!

php擴(kuò)展返回錯(cuò)誤的設(shè)置方法:1、打開(kāi)相應(yīng)的PHP文件;2、在擴(kuò)展中通過(guò)“php_error_docref()”函數(shù)來(lái)拋出錯(cuò)誤提示即可。

本文操作環(huán)境:windows7系統(tǒng)、PHP7.1版,DELL G3電腦

php 擴(kuò)展錯(cuò)誤怎么返回?

php源碼-擴(kuò)展中拋出和處理錯(cuò)誤

先說(shuō)說(shuō)源碼層面的錯(cuò)誤種類,大概有下面幾種

//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, 這種錯(cuò)誤會(huì)觸發(fā)try catch 異常處理流程,也就是會(huì)中斷當(dāng)前request的執(zhí)行,發(fā)生這種錯(cuò)誤時(shí)會(huì)把當(dāng)前要執(zhí)行的opcode設(shè)置為 ZEND_HANDLE_EXCEPTION, 從而跳出程序執(zhí)行 ,具體跳出過(guò)程參考: php源碼-異常throw處理過(guò)程-02

我們?cè)趯憯U(kuò)展的時(shí)候如何拋出錯(cuò)誤提示呢?

擴(kuò)展中可以通過(guò) php_error_docref()函數(shù)來(lái)拋出錯(cuò)誤提示,比如

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;
    }
}

通過(guò)php_error_docref這是的錯(cuò)誤提示,如果觸發(fā)的時(shí)候會(huì)出現(xiàn)類型下面的錯(cuò)誤提示輸出

相信在寫php代碼的時(shí)候也都見(jiàn)過(guò)這樣類似的錯(cuò)誤提示輸出

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

這個(gè)錯(cuò)誤提示輸出是如何實(shí)現(xiàn)的呢?

跟進(jìn)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:
                                //嚴(yán)重錯(cuò)誤,通過(guò)ZEND_HANDLE_EXCEPTION 中斷程序
                if (ex && ex->opline->opcode == ZEND_HANDLE_EXCEPTION &&
                    EG(opline_before_exception)) {
                    opline = EG(opline_before_exception);
                }
                break;
        }
    }
      //...
   // zend_error_cb 很重要, 這個(gè)函數(shù)是在 sapi啟動(dòng)的時(shí)候,通過(guò) php_module_startup() 賦值為 php_error_cb() 函數(shù)
   zend_error_cb(type, error_filename, error_lineno, format, args);
}

zend_error_cb 很重要, 這個(gè)函數(shù)是在 sapi啟動(dòng)的時(shí)候,通過(guò) php_module_startup() 賦值為 php_error_cb()函數(shù) , 而php_error_cb() 最終會(huì)調(diào)用 _sapi_module_struct.log_message(), 也就是當(dāng)你調(diào)用php_error_docref()函數(shù)來(lái)拋出錯(cuò)誤的時(shí)候,實(shí)際上會(huì)回調(diào) _sapi_module_struct.log_message (具體執(zhí)行過(guò)程參考:php源碼-sapi中自定義錯(cuò)誤輸出), 比如fpm sapi就會(huì)把錯(cuò)誤信息返回給, cli sapi通過(guò)自定義這個(gè)函數(shù)把錯(cuò)誤信息輸出到標(biāo)準(zhǔn)輸出

以上是“php中擴(kuò)展錯(cuò)誤如何返回”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對(duì)大家有幫助,更多相關(guān)知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

php
AI