亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

類型提示的實現

2019-11-15 02:05:39
字體:
來源:轉載
供稿:網友
類型提示的實現

php是弱類型語言,向方法傳遞參數時候也并不嚴格檢查數據類型。 不過有時需要判斷傳遞到方法中的參數,為此PHP中提供了一些函數,來判斷數據的類型。 比如is_numeric(),判斷是否是一個數值或者可轉換為數值的字符串,比如用于判斷對象的類型運算符:instanceof。 instanceof 用來測定一個給定的對象是否來自指定的對象類。instanceof 運算符是 PHP 5 引進的。 在此之前是使用的is_a(),不過現在已經不推薦使用。為了避免對象類型不規范引起的問題,PHP5中引入了類型提示這個概念。在定義方法參數時,同時定義參數的對象類型。 如果在調用的時候,傳入參數的類型與定義的參數類型不符,則會報錯。這樣就可以過濾對象的類型,或者說保證了數據的安全性。PHP中的類型提示功能只能用于參數為對象的提示,而無法用于為整數,字串,浮點等類型提示。在PHP5.1之后,PHP支持對數組的類型提示。要使用類型提示,只要在方法(或函數)的對象型參數前加一個已存在的類的名稱,當使用類型提示時, 你不僅可以指定對象類型,還可以指定抽象類和接口。一個數組的類型提示示例:

function array_PRint(Array $arr) {print_r($arr);}array_print(1);

以上的這段代碼有一點問題,它觸發了我們這次所介紹的類型提示,這段代碼在PHP5.1之后的版本執行,會報錯如下:

Catchable fatal error: Argument 1 passed to array_print() must be an array, integer given, called in  ...

當我們把函數參數中的整形變量變為數組時,程序會正常運行,調用print_r函數輸出數組。 那么這個類型提示是如何實現的呢? 不管是在類中的方法,還是我們調用的函數,都是使用function關鍵字作為其聲明的標記, 而類型提示的實現是與函數的聲明相關的,在聲明時就已經確定了參數的類型是哪些,但是需要在調用時才會顯示出來。 這里,我們從兩個方面說明類型提示的實現:

  1. 參數聲明時的類型提示
  2. 函數或方法調用時的類型提示

將剛才的那個例子修改一下:

    function array_print(Array $arr = 1) {    print_r($arr);}    array_print(array(1));

這段代碼與前面的那個示例相比,函數的參數設置了一個默認值,但是這個默認值是一個整形變量, 它與參數給定的類型提示Array不一樣,因此,當我們運行這段代碼時會很快看到程序會報錯如下:

Fatal error: Default value for parameters with array type hint can only be an array or NULL

為什么為很快看到報錯呢? 因為默認值的檢測過程發生在中間代碼生成階段,與運行時的報錯不同,它還沒有生成中間代碼,也沒有執行中間代碼的過程。 在Zend/zend_language_parser.y文件中,我們找到函數的參數列表在編譯時都會調用zend_do_receive_arg函數。 而在這個函數的參數列表中,第5個參數( znode *class_type)與我們這節所要表述的類型提示密切相關。 這個參數的作用是聲明類型提示中的類型,這里的類型有三種:

  1. 空,即沒有類型提示
  2. 類名,用戶定義或PHP自定義的類、接口等
  3. 數組,編譯期間對應的token是T_ARRAY,即Array字符串

在zend_do_receive_arg函數中,針對class_type參數做了一系列的操作,基本上是針對上面列出的三種類型, 其中對于類名,程序并沒有判斷這個類是否存在,即使你使用了一個不存在的類名, 程序在報錯時,顯示的也會是實參所給的對象并不是給定類的實例。以上是聲明類型提示的過程以及在聲明過程中對參數默認值的判斷過程,下面我們看下在函數或方法調用時類型提示的實現。從上面的聲明過程我們知道PHP在編譯類型提示的相關代碼時調用的是Zend/zend_complie.c文件中的zend_do_receive_arg函數, 在這個函數中將類型提示的判斷的opcode被賦值為ZEND_RECV。根據opcode的映射計算規則得出其在執行時調用的是ZEND_RECV_SPEC_HANDLER。 其代碼如下:

    static int ZEND_FASTCALL  ZEND_RECV_SPEC_HANDLER(ZEND_OPCODE_HANDLER_ARGS){        ...//省略        if (param == NULL) {                char *space;                char *class_name = get_active_class_name(&space TSRMLS_CC);                zend_execute_data *ptr = EX(prev_execute_data);             if (zend_verify_arg_type((zend_function *) EG(active_op_array), arg_num, NULL, opline->extended_value TSRMLS_CC)) {                   ...//省略            }               ...//省略            } else {              ...//省略                zend_verify_arg_type((zend_function *) EG(active_op_array), arg_num, *param, opline->extended_value TSRMLS_CC);              ...//省略        }  ...//省略}

如上所示:在ZEND_RECV_SPEC_HANDLER中最后調用的是zend_verify_arg_type。其代碼如下:

    static inline int zend_verify_arg_type(zend_function *zf, zend_uint arg_num, zval *arg, ulong fetch_type TSRMLS_DC){       ...//省略if (cur_arg_info->class_name) {    const char *class_name;    if (!arg) {        need_msg = zend_verify_arg_class_kind(cur_arg_info, fetch_type, &class_name, &ce TSRMLS_CC);        return zend_verify_arg_error(zf, arg_num, cur_arg_info, need_msg, class_name, "none", "" TSRMLS_CC);    }    if (Z_TYPE_P(arg) == IS_OBJECT) { // 既然是類對象參數, 傳遞的參數需要是對象類型        // 下面檢查這個對象是否是參數提示類的實例對象, 這里是允許傳遞子類實例對象        need_msg = zend_verify_arg_class_kind(cur_arg_info, fetch_type, &class_name, &ce TSRMLS_CC);        if (!ce || !instanceof_function(Z_OBJCE_P(arg), ce TSRMLS_CC)) {            return zend_verify_arg_error(zf, arg_num, cur_arg_info, need_msg, class_name, "instance of ", Z_OBJCE_P(arg)->name TSRMLS_CC);        }    } else if (Z_TYPE_P(arg) != IS_NULL || !cur_arg_info->allow_null) { // 參數為NULL, 也是可以通過檢查的,                                                                        // 如果函數定義了參數默認值, 不傳遞參數調用也是可以通過檢查的        need_msg = zend_verify_arg_class_kind(cur_arg_info, fetch_type, &class_name, &ce TSRMLS_CC);        return zend_verify_arg_error(zf, arg_num, cur_arg_info, need_msg, class_name, zend_zval_type_name(arg), "" TSRMLS_CC);    }    } else if (cur_arg_info->array_type_hint) { //  數組        if (!arg) {            return zend_verify_arg_error(zf, arg_num, cur_arg_info, "be an array", "", "none", "" TSRMLS_CC);        }        if (Z_TYPE_P(arg) != IS_ARRAY && (Z_TYPE_P(arg) != IS_NULL || !cur_arg_info->allow_null)) {            return zend_verify_arg_error(zf, arg_num, cur_arg_info, "be an array", "", zend_zval_type_name(arg), "" TSRMLS_CC);        }    }    return 1;}

zend_verify_arg_type的整個流程如圖3.1所示:

圖3.1 類型提示判斷流程圖

如果類型提示報錯,zend_verify_arg_type函數最后都會調用 zend_verify_arg_class_kind 生成報錯信息, 并且調用 zend_verify_arg_error 報錯。如下所示代碼:

static inline char * zend_verify_arg_class_kind(const zend_arg_info *cur_arg_info, ulong fetch_type, const char **class_name, zend_class_entry **pce TSRMLS_DC){    *pce = zend_fetch_class(cur_arg_info->class_name, cur_arg_info->class_name_len, (fetch_type | ZEND_FETCH_CLASS_AUTO | ZEND_FETCH_CLASS_NO_AUTOLOAD) TSRMLS_CC);*class_name = (*pce) ? (*pce)->name: cur_arg_info->class_name;if (*pce && (*pce)->ce_flags & ZEND_ACC_INTERFACE) {    return "implement interface ";} else {    return "be an instance of ";}}static inline int zend_verify_arg_error(const zend_function *zf, zend_uint arg_num, const zend_arg_info *cur_arg_info, const char *need_msg, const char *need_kind, const char *given_msg, char *given_kind TSRMLS_DC){    zend_execute_data *ptr = EG(current_execute_data)->prev_execute_data;    char *fname = zf->common.function_name;    char *fsep;    char *fclass;if (zf->common.scope) {    fsep =  "::";    fclass = zf->common.scope->name;} else {    fsep =  "";    fclass = "";}if (ptr && ptr->op_array) {    zend_error(E_RECOVERABLE_ERROR, "Argument %d passed to %s%s%s() must %s%s, %s%s given, called in %s on line %d and defined", arg_num, fclass, fsep, fname, need_msg, need_kind, given_msg, given_kind, ptr->op_array->filename, ptr->opline->lineno);} else {    zend_error(E_RECOVERABLE_ERROR, "Argument %d passed to %s%s%s() must %s%s, %s%s given", arg_num, fclass, fsep, fname, need_msg, need_kind, given_msg, given_kind);}return 0;}

在上面的代碼中,我們可以找到前面的報錯信息中的一些關鍵字Argument、 passed to、called in等。 這就是我們在調用函數或方法時類型提示顯示錯誤信息的最終執行位置。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
91嫩草在线视频| 高潮白浆女日韩av免费看| 欧美一区二区三区……| 亚洲美女av网站| 亚洲国产另类久久精品| 免费成人高清视频| 蜜臀久久99精品久久久久久宅男| 欧美激情视频给我| 欧美成人精品在线播放| 亚洲国产美女久久久久| 欧美大成色www永久网站婷| 国产精品户外野外| 国产有码在线一区二区视频| 国产精品一二区| 日韩欧美国产中文字幕| 欧美午夜精品久久久久久浪潮| 亚洲视频网站在线观看| 红桃视频成人在线观看| 日韩精品视频在线观看网址| 国产精品视频久久久久| 亚洲欧洲日本专区| 91精品国产一区| 亚洲护士老师的毛茸茸最新章节| 777777777亚洲妇女| 国产成人综合一区二区三区| 亚洲精品美女久久久久| 中文字幕在线看视频国产欧美在线看完整| 国产第一区电影| 91香蕉国产在线观看| 热久久免费视频精品| 456亚洲影院| 日韩二区三区在线| 欧美最猛黑人xxxx黑人猛叫黄| 色悠久久久久综合先锋影音下载| 久久久久久久久久久亚洲| 欧美一级黄色网| 日韩欧美中文字幕在线观看| 国产婷婷97碰碰久久人人蜜臀| 欧美日韩国产区| 91九色蝌蚪国产| 国内精久久久久久久久久人| 欧美做爰性生交视频| 尤物yw午夜国产精品视频明星| 国产精品久久久久久久久久久久| 精品国产成人在线| 亚洲精品mp4| 国产精品久久不能| 国产91色在线免费| 日韩欧美大尺度| 国产精品海角社区在线观看| 国内精品一区二区三区四区| 亚洲日本成人网| 色偷偷偷亚洲综合网另类| 久久久久在线观看| www.欧美三级电影.com| 日韩麻豆第一页| 日韩高清免费在线| 在线视频日韩精品| 国产精品日韩在线观看| 亚洲高清久久久久久| 日韩av免费在线播放| 午夜精品一区二区三区在线视频| 国产九九精品视频| 欧美在线视频观看免费网站| 国模吧一区二区三区| 成人黄色在线播放| 国产亚洲免费的视频看| 亚洲第一区第二区| 欧美日韩国产精品| 欧美性猛交xxxx富婆| 精品久久久久久国产91| 欧美日韩免费在线观看| 91精品国产91久久久久久久久| 亚洲精品成人久久电影| 精品性高朝久久久久久久| 福利一区视频在线观看| 成人国产精品日本在线| 色天天综合狠狠色| 亚洲欧美一区二区三区四区| 性色av一区二区三区红粉影视| 日韩在线播放av| 国产精品欧美激情在线播放| 欧美美女操人视频| 国产乱肥老妇国产一区二| 国产日韩精品在线| www.久久草.com| 九九九热精品免费视频观看网站| 国产成人自拍视频在线观看| 97久久伊人激情网| 亚洲国产精品久久久久| 国产精品美女www爽爽爽视频| 日韩电影免费观看中文字幕| 国产精品视频播放| 成人免费网视频| 日韩成人在线观看| 精品视频久久久久久久| 国产欧美精品一区二区三区介绍| 97精品一区二区视频在线观看| 色www亚洲国产张柏芝| 久久99精品久久久久久噜噜| 国产精品无码专区在线观看| 欧美日韩国产色| 欧美日韩在线视频首页| 91亚洲国产成人精品性色| 久久久精品在线观看| 国产欧美久久一区二区| 中文在线资源观看视频网站免费不卡| 精品一区电影国产| 国产精品白丝av嫩草影院| 日韩毛片在线看| 韩国精品久久久999| 欧美三级xxx| 亚洲国产精品va在线看黑人动漫| 国产亚洲欧洲高清| 91在线中文字幕| 亚洲人成电影网站色…| 国产精品香蕉国产| 国产视频亚洲视频| 国产精品视频网| 成人黄色中文字幕| 久久久久久亚洲精品中文字幕| 国产剧情久久久久久| 欧美精品videofree1080p| 美女视频黄免费的亚洲男人天堂| 在线观看国产精品91| 欧美一级视频在线观看| 91久热免费在线视频| 日韩激情av在线播放| 九九精品视频在线观看| 原创国产精品91| 91色视频在线观看| 精品国产自在精品国产浪潮| 欧美日韩综合视频| 久久久在线观看| 91黑丝高跟在线| 欧美色xxxx| 国产亚洲视频在线观看| 秋霞成人午夜鲁丝一区二区三区| 国产精品福利小视频| 亚洲国产精品成人av| 日韩精品视频在线观看网址| 成人av色在线观看| 亚洲美女自拍视频| 欧美日韩中文在线| 久99九色视频在线观看| 夜夜嗨av一区二区三区免费区| 国产成人极品视频| 国产福利视频一区二区| 国产精品第一第二| 97精品在线视频| 欧美亚洲视频在线看网址| 69国产精品成人在线播放| 欧美激情a在线| 在线精品国产成人综合| 欧美激情欧美狂野欧美精品| 欧美人交a欧美精品| 亚洲精品videossex少妇| 中文字幕日韩欧美在线| 在线播放日韩欧美| 国产精品色午夜在线观看| 九九久久精品一区| 亚洲网站视频福利| 免费99精品国产自在在线| 成人久久久久爱|