重載和重寫
先區(qū)分一下重載(overload)和重寫(override):重載指多個名字相同,但參數不同的函數在同一作用域并存的現象;重寫出現在繼承中,指子類重定義父類功能的現象,也被稱為覆蓋。重載中說的參數不同有三種情況:參數個數不同,參數類型不同,參數順序不同。重寫一般指函數的覆蓋,即相同簽名的成員函數在子類中重新定義(實現抽象函數或接口不是重寫),是實現多態(tài)(polymorphism)的一種關鍵技術。成員變量也可以重載/覆蓋,但一般不會這么做。
用簡單的C代碼來說明重載:
int add(int a, int b) { return a + b; }
double add(double a, double b) { return a + b; }
double add(int a, int b, double c) { return a + b + c; }
double add(double a, int b, int c) { return a + b + c; }
第一個函數為參考基準,其他三個對應重載的三種情形。函數重載多見于強類型語言,編譯后函數在函數符號表的名稱一般是函數名加參數類型。上面的四個函數,g++編譯后,nm命令查看符號表中的名字,輸出如下:
[tlanyan@server ~]# nm test | grep add
0000000000400730 t _GLOBAL__sub_I__Z3addii
0000000000400851 T _Z3adddd
00000000004008b1 T _Z3adddii
000000000040083d T _Z3addii
000000000040087d T _Z3addiid
最后四行的第三列對應編譯后四個函數的符號信息,_Z3為前綴,add是函數名,剩下的字母d代表double,i代表int,與生命一一對應。
再回到PHP的重載。PHP的函數聲明中參數無需聲明類型,直接排除參數類型不同、參數順序不同兩種重載,只剩下參數個數不同一條路可走。定義一個參數個數不同名字相同的函數,這么一個小小的重載要求,在PHP中也是不合法的!原因是PHP中不允許同名函數存在,想定義重名函數,死心吧!雖然大多數情況下以默認參數方式實現重載基本上夠用,但不時還會覺得憋屈,忍不住想問一句:PHP為什么不允許(同名函數)重載啊?!
PHP的苦衷
PHP不支持同名函數的重載是有原因的。上面已經提到,PHP函數聲明時不需要指定參數類型,重載中的三種情況立馬廢掉兩種。幸存的參數個數不同這一條路也走不通,為什么呢?因為PHP中調用函數時,少傳參數,不行;多傳參數,沒問題!來個簡單的例子:
function foo($arg1, $arg2) {
echo "$arg1, $arg2\n";
}
// 函數調用
// 參數過少,提示:
//PHP Warning: Missing argument 2 for foo()
// PHP Notice: Undefined variable: arg2 in php shell code on line 2
foo("tlanyan");
// 參數個數正好,運行正常
foo("hello", "tlanyan");
// 多傳參數,運行正常
foo("hello", "tlanyan", "nice day");
// 傳更多參數,也一切正常
foo("hello", "tlanyan", "morning", "noon", "afternoon", "evening", "night");
只要個數不小于聲明的,傳多少參數PHP不管。所以參數個數不同,在PHP中不足以區(qū)分函數。
個人認為另一個不允許名函數存在的重要原因是function_exists
、 method_exists
、is_callable
這些API的存在。作為簡單易用的語言,PHP為開發(fā)人員提供了查詢函數名是否存在/可用的便利API,這在編程語言中很少見(尤其是get_defined_functions
這類API)??梢钥吹?,這些API都不需要參數信息。如果能定義參數不同的重載函數,這些API都要跟著改,勢必引入額外的復雜性。正所謂魚與熊掌不可兼得,方便你用時沒想到參數不同,不方便你定義就抱怨,好像不好吧?
PHP5引入了反射API,這是非常強大的類型信息查詢工具。就函數聲明而言,ReflectionMethod/ReflectionFunction類的getParameters/getNumberOfParameters/getNumberOfRequiredParameters等API,功能上甩function_exists
等好幾條街。有了反射機制,按理說function_exists
這些API可以安心的退休。雖然反射這一套東西功能強大,但遠沒有舊式API簡單好用。再加上看看市面上的代碼,有多少類庫和框架依賴舊式API。從兼容性和實用性考慮,個人認為短時間內能以同名函數方式重載的概率非常小。
PHP中的重載
只看完上面的內容就說PHP不支持重載,我想隨便一個資深的PHP開發(fā)都會不由自主的取下拖鞋,然后教你什么是PHP中的重載,并保證至少有好幾種實現方法;官方人員對這種認知估計也無力吐槽:能不能好好看官方文檔?!官網中可是有一節(jié)專門講重載!
因為種種原因,PHP不支持傳統的重載,也就是同名函數的重載,但PHP是支持重載的,而且姿勢還不少。簡單來說,PHP中主要有以下幾種重載方式:
- 默認參數,定義一個全面的函數版本,不是必須的值在聲明時賦予默認值;
- 定義一個不聲明參數的入口函數,函數內使用func_num_args/func_get_args獲取參數個數/數組,然后根據參數個數轉發(fā)到具體實現的函數;
- 自PHP5.6起,可以用變長參數實現重載,func_get_args的另一種形式;
- 對于類中的成員函數,可以通過__call和__callStatic實現重載。
如果你還知道其他方式,歡迎評論給出方案。
總結
PHP的特性決定了其不支持同名函數方式的重載,但并不意味著PHP不支持重載。實際上PHP可以多種方式實現重載,并保持其一貫的簡單易用性。
感謝閱讀!
以上就是PHP重載基礎知識回顧的詳細內容,更多關于PHP 重載的資料請關注腳本之家其它相關文章!
您可能感興趣的文章:- 解決PHP Opcache 緩存刷新、代碼重載出現無法更新代碼的問題
- php 使用 __call實現重載功能示例
- PHP面向對象程序設計模擬一般面向對象語言中的方法重載(overload)示例
- PHP面相對象中的重載與重寫
- PHP中子類重載父類的方法【parent::方法名】
- PHP面向對象編程之深入理解方法重載與方法覆蓋(多態(tài))
- php函數重載的替代方法--偽重載詳解
- php繼承中方法重載(覆蓋)的應用場合
- PHP使用方法重載實現動態(tài)創(chuàng)建屬性的get和set方法
- PHP利用func_get_args和func_num_args函數實現函數重載實例
- php面向對象全攻略 (八)重載新的方法
- php面向對象的方法重載兩種版本比較