摘要:從兼容性和實(shí)用性考慮,個(gè)人認(rèn)為短時(shí)間內(nèi)能以同名函數(shù)方式重載的概率非常小??偨Y(jié)的特性決定了其不支持同名函數(shù)方式的重載,但并不意味著不支持重載。
轉(zhuǎn)載請(qǐng)注明文章出處:https://tlanyan.me/overload-i...
整理思路時(shí)想到一個(gè)問(wèn)題:PHP為什么不允許同名函數(shù)存在?即不允許常見(jiàn)于其他語(yǔ)言的重載機(jī)制?
重載和重寫(xiě)先區(qū)分一下重載(overload)和重寫(xiě)(override):重載指多個(gè)名字相同,但參數(shù)不同的函數(shù)在同一作用域并存的現(xiàn)象;重寫(xiě)出現(xiàn)在繼承中,指子類重定義父類功能的現(xiàn)象,也被稱為覆蓋。重載中說(shuō)的參數(shù)不同有三種情況:參數(shù)個(gè)數(shù)不同,參數(shù)類型不同,參數(shù)順序不同。重寫(xiě)一般指函數(shù)的覆蓋,即相同簽名的成員函數(shù)在子類中重新定義(實(shí)現(xiàn)抽象函數(shù)或接口不是重寫(xiě)),是實(shí)現(xiàn)多態(tài)(polymorphism)的一種關(guān)鍵技術(shù)。成員變量也可以重載/覆蓋,但一般不會(huì)這么做。
用簡(jiǎn)單的C代碼來(lái)說(shuō)明重載:
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ù)重載多見(jiàn)于強(qiáng)類型語(yǔ)言,編譯后函數(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 _Z3affffdd 00000000004008b1 T _Z3affffdii 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ù)無(wú)需聲明類型,直接排除參數(shù)類型不同、參數(shù)順序不同兩種重載,只剩下參數(shù)個(gè)數(shù)不同一條路可走。定義一個(gè)參數(shù)個(gè)數(shù)不同名字相同的函數(shù),這么一個(gè)小小的重載要求,在PHP中也是不合法的!原因是PHP中不允許同名函數(shù)存在,想定義重名函數(shù),死心吧!雖然大多數(shù)情況下以默認(rèn)參數(shù)方式實(shí)現(xiàn)重載基本上夠用,但不時(shí)還會(huì)覺(jué)得憋屈,忍不住想問(wèn)一句:PHP為什么不允許(同名函數(shù))重載???!
PHP的苦衷PHP不支持同名函數(shù)的重載是有原因的。上面已經(jīng)提到,PHP函數(shù)聲明時(shí)不需要指定參數(shù)類型,重載中的三種情況立馬廢掉兩種。幸存的參數(shù)個(gè)數(shù)不同這一條路也走不通,為什么呢?因?yàn)镻HP中調(diào)用函數(shù)時(shí),少傳參數(shù),不行;多傳參數(shù),沒(méi)問(wèn)題!來(lái)個(gè)簡(jiǎn)單的例子:
function foo($arg1, $arg2) { echo "$arg1, $arg2 "; } // 函數(shù)調(diào)用 // 參數(shù)過(guò)少,提示: //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)單易用的語(yǔ)言,PHP為開(kāi)發(fā)人員提供了查詢函數(shù)名是否存在/可用的便利API,這在編程語(yǔ)言中很少見(jiàn)(尤其是get_defined_functions這類API)??梢钥吹剑@些API都不需要參數(shù)信息。如果能定義參數(shù)不同的重載函數(shù),這些API都要跟著改,勢(shì)必引入額外的復(fù)雜性。正所謂魚(yú)與熊掌不可兼得,方便你用時(shí)沒(méi)想到參數(shù)不同,不方便你定義就抱怨,好像不好吧?
PHP5引入了反射API,這是非常強(qiáng)大的類型信息查詢工具。就函數(shù)聲明而言,ReflectionMethod/ReflectionFunction類的getParameters/getNumberOfParameters/getNumberOfRequiredParameters等API,功能上甩function_exists等好幾條街。有了反射機(jī)制,按理說(shuō)function_exists這些API可以安心的退休。雖然反射這一套東西功能強(qiáng)大,但遠(yuǎn)沒(méi)有舊式API簡(jiǎn)單好用。再加上看看市面上的代碼,有多少類庫(kù)和框架依賴舊式API。從兼容性和實(shí)用性考慮,個(gè)人認(rèn)為短時(shí)間內(nèi)能以同名函數(shù)方式重載的概率非常小。
PHP中的重載只看完上面的內(nèi)容就說(shuō)PHP不支持重載,我想隨便一個(gè)資深的PHP開(kāi)發(fā)都會(huì)不由自主的取下拖鞋,然后教你什么是PHP中的重載,并保證至少有好幾種實(shí)現(xiàn)方法;官方人員對(duì)這種認(rèn)知估計(jì)也無(wú)力吐槽:能不能好好看官方文檔?!官網(wǎng)中可是有一節(jié)專門(mén)講重載!
因?yàn)榉N種原因,PHP不支持傳統(tǒng)的重載,也就是同名函數(shù)的重載,但PHP是支持重載的,而且姿勢(shì)還不少。簡(jiǎn)單來(lái)說(shuō),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ù),可以通過(guò)__call和__callStatic實(shí)現(xiàn)重載。
如果你還知道其他方式,歡迎評(píng)論給出方案。
總結(jié)PHP的特性決定了其不支持同名函數(shù)方式的重載,但并不意味著PHP不支持重載。實(shí)際上PHP可以多種方式實(shí)現(xiàn)重載,并保持其一貫的簡(jiǎn)單易用性。
感謝閱讀!
參考http://php.net/manual/it/lang...
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/30819.html
摘要:重載在中就大量應(yīng)用了重載相關(guān)知識(shí),如在中就用到了方法重載知識(shí)使用魔術(shù)方法來(lái)動(dòng)態(tài)創(chuàng)建類中未定義或不可見(jiàn)的靜態(tài)方法。中通過(guò)引入魔術(shù)方法來(lái)實(shí)現(xiàn)動(dòng)態(tài)的創(chuàng)建類屬性和方法,包括屬性重載的魔術(shù)方法和方法重載的魔術(shù)方法。 說(shuō)明:本文主要講述PHP中重載概念,由于Laravel框架中經(jīng)常使用這塊知識(shí)點(diǎn),并且PHP的重載概念又與其他OOP語(yǔ)言如JAVA中重載概念不一樣,故復(fù)習(xí)并記錄相關(guān)知識(shí)點(diǎn)。同時(shí),作者會(huì)...
摘要:轉(zhuǎn)載自我的博客在眾多的新特性中,我覺(jué)得這是最神奇甚至是詭異的一個(gè),如果有不理解這個(gè)概念的朋友,可能連它的說(shuō)明都看不懂。在這個(gè)例子中操作符被重載為,操作符被重載為。再重載成操作符,以后咋跟蹤代碼。。。 轉(zhuǎn)載自我的博客:http://70.io/2014/03/php-5_6-internal-operator-overloading 在眾多php 5.6的新特性中,我覺(jué)得這是最...
摘要:魔術(shù)方法知識(shí)點(diǎn)整理代碼使用語(yǔ)法編寫(xiě)一構(gòu)造函數(shù)和析構(gòu)函數(shù)構(gòu)造函數(shù)具有構(gòu)造函數(shù)的類會(huì)在每次創(chuàng)建新對(duì)象時(shí)先調(diào)用此方法,所以非常適合在使用對(duì)象之前做一些初始化工作。在析構(gòu)函數(shù)中調(diào)用將會(huì)中止其余關(guān)閉操作的運(yùn)行。析構(gòu)函數(shù)中拋異常會(huì)導(dǎo)致致命錯(cuò)誤。 PHP魔術(shù)方法知識(shí)點(diǎn)整理 代碼使用PHP7.2語(yǔ)法編寫(xiě) 一、構(gòu)造函數(shù)和析構(gòu)函數(shù) __construct() 構(gòu)造函數(shù) __construct ([ mi...
摘要:類與對(duì)象基本概念如果在之后跟著的是一個(gè)包含有類名的字符串,則該類的一個(gè)實(shí)例被創(chuàng)建。如果該類屬于一個(gè)名字空間,則必須使用其完整名稱。如果一個(gè)類被聲明為,則不能被繼承。命名空間通過(guò)關(guān)鍵字來(lái)聲明。 類與對(duì)象 基本概念 new:如果在 new 之后跟著的是一個(gè)包含有類名的字符串,則該類的一個(gè)實(shí)例被創(chuàng)建。如果該類屬于一個(gè)名字空間,則必須使用其完整名稱。 Example #3 創(chuàng)建一個(gè)實(shí)例 ...
摘要:魔術(shù)方法構(gòu)造函數(shù)允行開(kāi)發(fā)者在一個(gè)類中定義一個(gè)方法作為構(gòu)造函數(shù)。如果子類沒(méi)有定義構(gòu)造函數(shù)則會(huì)如同一個(gè)普通的類方法一樣從父類繼承假如沒(méi)有被定義為的話。和函數(shù)會(huì)檢查類中是否存在一個(gè)魔術(shù)方法。魔術(shù)常量魔法方法 PHP - 魔術(shù)常量 __LINE__ 文件中的當(dāng)前行號(hào) __FILE__ 文件的完整路徑和文件名。如果用在被包含文件中,則返回被包含的文件名。 __DIR__ 文件所在的目錄。如果用在...
閱讀 2207·2023-04-25 15:00
閱讀 2388·2021-11-18 13:14
閱讀 1245·2021-11-15 11:37
閱讀 3130·2021-09-24 13:55
閱讀 1256·2019-08-30 15:52
閱讀 2672·2019-08-29 12:35
閱讀 3389·2019-08-29 11:04
閱讀 1237·2019-08-26 12:13