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