PHP是弱類型語言,向方法傳遞參數時候也并不嚴格檢查數據類型。不過有時需要判斷傳遞到方法中的參數,為此PHP中提供了一些函數,來判斷數據的類型。比如is_numeric(),判斷是否是一個數值或者可轉換為數值的字符串,比如用于判斷對象的類型運算符:instanceof。instanceof 用來測定一個給定的對象是否來自指定的對象類。instanceof 運算符是 PHP 5 引進的。在此之前是使用的is_a(),不過現在已經不推薦使用。
為了避免對象類型不規范引起的問題,PHP5中引入了類型提示這個概念。在定義方法參數時,同時定義參數的對象類型。如果在調用的時候,傳入參數的類型與定義的參數類型不符,則會報錯。這樣就可以過濾對象的類型,或者說保證了數據的安全性。
> PHP中的類型提示功能只能用于參數為對象的提示,而無法用于為整數,字串,浮點等類型提示。在PHP5.1之后,PHP支持對數組的類型提示。
要使用類型提示,只要在方法(或函數)的對象型參數前加一個已存在的類的名稱,當使用類型提示時,你不僅可以指定對象類型,還可以指定抽象類和接口。
一個數組的類型提示示例:
function array_print([Array](http://www.php.net/array) $arr) {
[print_r](http://www.php.net/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. 參數聲明時的類型提示
1. 函數或方法調用時的類型提示
將剛才的那個例子修改一下:
function array_print([Array](http://www.php.net/array) $arr = 1) {
[print_r](http://www.php.net/print_r)($arr);
}
?
array_print([array](http://www.php.net/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. 空,即沒有類型提示
1. 類名,用戶定義或PHP自定義的類、接口等
1. 數組,編譯期間對應的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等。這就是我們在調用函數或方法時類型提示顯示錯誤信息的最終執行位置。
- 第一章 準備工作和背景知識
- 第一節 環境搭建
- 第二節 源碼結構、閱讀代碼方法
- 第三節 常用代碼
- 第四節 小結
- 第二章 用戶代碼的執行
- 第一節 生命周期和Zend引擎
- 第二節 SAPI概述
- Apache模塊
- 嵌入式
- FastCGI
- 第三節 PHP腳本的執行
- 詞法分析和語法分析
- opcode
- opcode處理函數查找
- 第四節 小結
- 第三章 變量及數據類型
- 第一節 變量的結構和類型
- 哈希表(HashTable)
- PHP的哈希表實現
- 鏈表簡介
- 第二節 常量
- 第三節 預定義變量
- 第四節 靜態變量
- 第五節 類型提示的實現
- 第六節 變量的生命周期
- 變量的賦值和銷毀
- 變量的作用域
- global語句
- 第七節 數據類型轉換
- 第八節 小結
- 第四章 函數的實現
- 第一節 函數的內部結構
- 函數的內部結構
- 函數間的轉換
- 第二節 函數的定義,傳參及返回值
- 函數的定義
- 函數的參數
- 函數的返回值
- 第三節 函數的調用和執行
- 第四節 匿名函數及閉包
- 第五節 小結
- 第五章 類和面向對象
- 第一節 類的結構和實現
- 第二節 類的成員變量及方法
- 第三節 訪問控制的實現
- 第四節 類的繼承,多態及抽象類
- 第五節 魔術方法,延遲綁定及靜態成員
- 第六節 PHP保留類及特殊類
- 第七節 對象
- 第八節 命名空間
- 第九節 標準類
- 第十節 小結
- 第六章 內存管理
- 第一節 內存管理概述
- 第二節 PHP中的內存管理
- 第三節 內存使用:申請和銷毀
- 第四節 垃圾回收
- 新的垃圾回收
- 第五節 內存管理中的緩存
- 第六節 寫時復制(Copy On Write)
- 第七節 內存泄漏
- 第八節 小結
- 第七章 Zend虛擬機
- 第一節 Zend虛擬機概述
- 第二節 語法的實現
- 詞法解析
- 語法分析
- 實現自己的語法
- 第三節 中間代碼的執行
- 第四節 PHP代碼的加密解密
- 第五節 小結
- 第八章 線程安全
- 第二節 線程,進程和并發
- 第三節 PHP中的線程安全
- 第九章 錯誤和異常處理
- 第十章 輸出緩沖
- 第十六章 PHP語言特性的實現
- 第一節 循環語句
- foreach的實現
- 第二十章 怎么樣系列(how to)
- 附錄
- 附錄A PHP及Zend API
- 附錄B PHP的歷史
- 附錄C VLD擴展使用指南
- 附錄D 怎樣為PHP貢獻
- 附錄E phpt測試文件說明
- 附錄F PHP5.4新功能升級解析
- 附錄G:re2c中文手冊