您好,登錄后才能下訂單哦!
這篇文章主要講解了“PHP重載的基礎知識”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“PHP重載的基礎知識”吧!
重載和重寫
先區分一下重載(overload)和重寫(override):重載指多個名字相同,但參數不同的函數在同一作用域并存的現象;重寫出現在繼承中,指子類重定義父類功能的現象,也被稱為覆蓋。重載中說的參數不同有三種情況:參數個數不同,參數類型不同,參數順序不同。重寫一般指函數的覆蓋,即相同簽名的成員函數在子類中重新定義(實現抽象函數或接口不是重寫),是實現多態(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中不足以區分函數。
個人認為另一個不允許名函數存在的重要原因是function_exists
、 method_exists
、is_callable
這些API的存在。作為簡單易用的語言,PHP為開發人員提供了查詢函數名是否存在/可用的便利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開發都會不由自主的取下拖鞋,然后教你什么是PHP中的重載,并保證至少有好幾種實現方法;官方人員對這種認知估計也無力吐槽:能不能好好看官方文檔?!官網中可是有一節專門講重載!
因為種種原因,PHP不支持傳統的重載,也就是同名函數的重載,但PHP是支持重載的,而且姿勢還不少。簡單來說,PHP中主要有以下幾種重載方式:
默認參數,定義一個全面的函數版本,不是必須的值在聲明時賦予默認值;
定義一個不聲明參數的入口函數,函數內使用func_num_args/func_get_args獲取參數個數/數組,然后根據參數個數轉發到具體實現的函數;
自PHP5.6起,可以用變長參數實現重載,func_get_args的另一種形式;
對于類中的成員函數,可以通過__call和__callStatic實現重載。
如果你還知道其他方式,歡迎評論給出方案。
總結
PHP的特性決定了其不支持同名函數方式的重載,但并不意味著PHP不支持重載。實際上PHP可以多種方式實現重載,并保持其一貫的簡單易用性。
感謝各位的閱讀,以上就是“PHP重載的基礎知識”的內容了,經過本文的學習后,相信大家對PHP重載的基礎知識這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。