域名預(yù)訂/競(jìng)價(jià),好“米”不錯(cuò)過
這篇文章主要介紹了PHP重載的相關(guān)知識(shí),幫助大家更好的理解和復(fù)習(xí)php,感興趣的朋友可以了解下
重載和重寫
先區(qū)分一下重載(overload)和重寫(override):重載指多個(gè)名字相同,但參數(shù)不同的函數(shù)在同一作用域并存的現(xiàn)象;重寫出現(xiàn)在繼承中,指子類重定義父類功能的現(xiàn)象,也被稱為覆蓋。重載中說的參數(shù)不同有三種情況:參數(shù)個(gè)數(shù)不同,參數(shù)類型不同,參數(shù)順序不同。重寫一般指函數(shù)的覆蓋,即相同簽名的成員函數(shù)在子類中重新定義(實(shí)現(xiàn)抽象函數(shù)或接口不是重寫),是實(shí)現(xiàn)多態(tài)(polymorphism)的一種關(guān)鍵技術(shù)。成員變量也可以重載/覆蓋,但一般不會(huì)這么做。
用簡(jiǎn)單的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è)函數(shù)為參考基準(zhǔn),其他三個(gè)對(duì)應(yīng)重載的三種情形。函數(shù)重載多見于強(qiáng)類型語言,編譯后函數(shù)在函數(shù)符號(hào)表的名稱一般是函數(shù)名加參數(shù)類型。上面的四個(gè)函數(shù),g++編譯后,nm命令查看符號(hào)表中的名字,輸出如下:
[tlanyan@server ~]# nm test | grep add
0000000000400730 t _GLOBAL__sub_I__Z3addii
0000000000400851 T _Z3adddd
00000000004008b1 T _Z3adddii
000000000040083d T _Z3addii
000000000040087d T _Z3addiid
最后四行的第三列對(duì)應(yīng)編譯后四個(gè)函數(shù)的符號(hào)信息,_Z3為前綴,add是函數(shù)名,剩下的字母d代表double,i代表int,與生命一一對(duì)應(yīng)。
再回到PHP的重載。PHP的函數(shù)聲明中參數(shù)無需聲明類型,直接排除參數(shù)類型不同、參數(shù)順序不同兩種重載,只剩下參數(shù)個(gè)數(shù)不同一條路可走。定義一個(gè)參數(shù)個(gè)數(shù)不同名字相同的函數(shù),這么一個(gè)小小的重載要求,在PHP中也是不合法的!原因是PHP中不允許同名函數(shù)存在,想定義重名函數(shù),死心吧!雖然大多數(shù)情況下以默認(rèn)參數(shù)方式實(shí)現(xiàn)重載基本上夠用,但不時(shí)還會(huì)覺得憋屈,忍不住想問一句:PHP為什么不允許(同名函數(shù))重載?。?!
PHP的苦衷
PHP不支持同名函數(shù)的重載是有原因的。上面已經(jīng)提到,PHP函數(shù)聲明時(shí)不需要指定參數(shù)類型,重載中的三種情況立馬廢掉兩種。幸存的參數(shù)個(gè)數(shù)不同這一條路也走不通,為什么呢?因?yàn)镻HP中調(diào)用函數(shù)時(shí),少傳參數(shù),不行;多傳參數(shù),沒問題!來個(gè)簡(jiǎn)單的例子:
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ù)個(gè)數(shù)正好,運(yùn)行正常
foo("hello", "tlanyan");
// 多傳參數(shù),運(yùn)行正常
foo("hello", "tlanyan", "nice day");
// 傳更多參數(shù),也一切正常
foo("hello", "tlanyan", "morning", "noon", "afternoon", "evening", "night");
只要個(gè)數(shù)不小于聲明的,傳多少參數(shù)PHP不管。所以參數(shù)個(gè)數(shù)不同,在PHP中不足以區(qū)分函數(shù)。
個(gè)人認(rèn)為另一個(gè)不允許名函數(shù)存在的重要原因是function_exists、 method_exists、is_callable這些API的存在。作為簡(jiǎn)單易用的語言,PHP為開發(fā)人員提供了查詢函數(shù)名是否存在/可用的便利API,這在編程語言中很少見(尤其是get_defined_functions這類API)??梢钥吹?,這些API都不需要參數(shù)信息。如果能定義參數(shù)不同的重載函數(shù),這些API都要跟著改,勢(shì)必引入額外的復(fù)雜性。正所謂魚與熊掌不可兼得,方便你用時(shí)沒想到參數(shù)不同,不方便你定義就抱怨,好像不好吧?
PHP5引入了反射API,這是非常強(qiáng)大的類型信息查詢工具。就函數(shù)聲明而言,ReflectionMethod/ReflectionFunction類的getParameters/getNumberOfParameters/getNumberOfRequiredParameters等API,功能上甩function_exists等好幾條街。有了反射機(jī)制,按理說function_exists這些API可以安心的退休。雖然反射這一套東西功能強(qiáng)大,但遠(yuǎn)沒有舊式API簡(jiǎn)單好用。再加上看看市面上的代碼,有多少類庫和框架依賴舊式API。從兼容性和實(shí)用性考慮,個(gè)人認(rèn)為短時(shí)間內(nèi)能以同名函數(shù)方式重載的概率非常小。
PHP中的重載
只看完上面的內(nèi)容就說PHP不支持重載,我想隨便一個(gè)資深的PHP開發(fā)都會(huì)不由自主的取下拖鞋,然后教你什么是PHP中的重載,并保證至少有好幾種實(shí)現(xiàn)方法;官方人員對(duì)這種認(rèn)知估計(jì)也無力吐槽:能不能好好看官方文檔?!官網(wǎng)中可是有一節(jié)專門講重載!
因?yàn)榉N種原因,PHP不支持傳統(tǒng)的重載,也就是同名函數(shù)的重載,但PHP是支持重載的,而且姿勢(shì)還不少。簡(jiǎn)單來說,PHP中主要有以下幾種重載方式:
默認(rèn)參數(shù),定義一個(gè)全面的函數(shù)版本,不是必須的值在聲明時(shí)賦予默認(rèn)值;
定義一個(gè)不聲明參數(shù)的入口函數(shù),函數(shù)內(nèi)使用func_num_args/func_get_args獲取參數(shù)個(gè)數(shù)/數(shù)組,然后根據(jù)參數(shù)個(gè)數(shù)轉(zhuǎn)發(fā)到具體實(shí)現(xiàn)的函數(shù);
自PHP5.6起,可以用變長(zhǎng)參數(shù)實(shí)現(xiàn)重載,func_get_args的另一種形式;
對(duì)于類中的成員函數(shù),可以通過__call和__callStatic實(shí)現(xiàn)重載。
如果你還知道其他方式,歡迎評(píng)論給出方案。
總結(jié)
PHP的特性決定了其不支持同名函數(shù)方式的重載,但并不意味著PHP不支持重載。實(shí)際上PHP可以多種方式實(shí)現(xiàn)重載,并保持其一貫的簡(jiǎn)單易用性。
來源:腳本之家
鏈接:https://www.jb51.net/article/195311.htm
申請(qǐng)創(chuàng)業(yè)報(bào)道,分享創(chuàng)業(yè)好點(diǎn)子。點(diǎn)擊此處,共同探討創(chuàng)業(yè)新機(jī)遇!