PHP重载基础知识回顾


Posted in PHP onSeptember 10, 2020

重载和重写

先区分一下重载(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_existsmethod_existsis_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中主要有以下几种重载方式:

  1. 默认参数,定义一个全面的函数版本,不是必须的值在声明时赋予默认值;
  2. 定义一个不声明参数的入口函数,函数内使用func_num_args/func_get_args获取参数个数/数组,然后根据参数个数转发到具体实现的函数;
  3. 自PHP5.6起,可以用变长参数实现重载,func_get_args的另一种形式;
  4. 对于类中的成员函数,可以通过__call和__callStatic实现重载。

如果你还知道其他方式,欢迎评论给出方案。

总结

PHP的特性决定了其不支持同名函数方式的重载,但并不意味着PHP不支持重载。实际上PHP可以多种方式实现重载,并保持其一贯的简单易用性。

感谢阅读!

以上就是PHP重载基础知识回顾的详细内容,更多关于PHP 重载的资料请关注三水点靠木其它相关文章!

PHP 相关文章推荐
一次编写,随处运行
Oct 09 PHP
PHP通用分页类page.php[仿google分页]
Aug 31 PHP
php中利用str_pad函数生成数字递增形式的产品编号
Sep 30 PHP
php面向对象中的魔术方法中文说明
Mar 04 PHP
PHP解码unicode编码的中文字符代码分享
Aug 13 PHP
PHP实现更新中间关联表数据的两种方法
Sep 01 PHP
PHP异常处理浅析
May 12 PHP
PHP常用的排序和查找算法
Aug 06 PHP
Session 失效的原因汇总及解决丢失办法
Sep 30 PHP
PHP多维数组遍历方法(2种实现方法)
Dec 10 PHP
详解PHP中cookie和session的区别及cookie和session用法小结
Jun 12 PHP
php版微信公众平台接口参数调试实现判断用户行为的方法
Sep 23 PHP
PHP反射基础知识回顾
Sep 10 #PHP
PHP获取类私有属性的3种方法
Sep 10 #PHP
php实现图片压缩处理
Sep 09 #PHP
如何在PHP中读写文件
Sep 07 #PHP
PHP延迟静态绑定使用方法实例解析
Sep 05 #PHP
PHP autoload使用方法及步骤详解
Sep 05 #PHP
PHP数组访问常用方法解析
Sep 05 #PHP
You might like
实用函数10
2007/11/08 PHP
PHP开发规范手册之PHP代码规范详解
2011/01/13 PHP
PHP数组排序函数合集 以及它们之间的联系分析
2013/06/27 PHP
PHP 文件编程综合案例-文件上传的实现
2013/07/03 PHP
PHP匿名函数和use子句用法实例
2016/03/16 PHP
php使用pclzip类实现文件压缩的方法(附pclzip类下载地址)
2016/04/30 PHP
通过源码解析Laravel的依赖注入
2018/01/22 PHP
JavaScript中的私有成员
2006/09/18 Javascript
身份证号码前六位所代表的省,市,区, 以及地区编码下载
2007/04/12 Javascript
js类后台管理菜单类-MenuSwitch
2007/09/12 Javascript
js类的静态属性和实例属性的理解
2009/10/01 Javascript
javascript 全等号运算符使用说明
2010/05/31 Javascript
理解Javascript_03_javascript全局观
2010/10/11 Javascript
jQuery之网页换肤实现代码
2011/04/30 Javascript
iframe异步加载实现点击左边菜单加载右边内容实例讲解
2013/03/04 Javascript
基于jquery异步传输json数据格式实例代码
2013/11/23 Javascript
js动态添加删除,后台取数据(示例代码)
2013/11/25 Javascript
使用JavaScript刷新网页的方法
2015/06/04 Javascript
jQuery添加options点击事件并传值实例代码
2016/05/18 Javascript
jQuery弹出遮罩层效果完整示例
2016/09/13 Javascript
分享一道关于闭包、bind和this的面试题
2017/02/20 Javascript
基于JQuery的购物车添加删除以及结算功能示例
2017/03/08 Javascript
vue-scroller记录滚动位置的示例代码
2018/01/17 Javascript
vuex state中的数组变化监听实例
2019/11/06 Javascript
vue移动端使用appClound拉起支付宝支付的实现方法
2019/11/21 Javascript
[01:07:20]DOTA2-DPC中国联赛 正赛 Dynasty vs XG BO3 第二场 2月2日
2021/03/11 DOTA
Python 私有函数的实例详解
2017/09/11 Python
tensorflow之并行读入数据详解
2020/02/05 Python
借助Paramiko通过Python实现linux远程登陆及sftp的操作
2020/03/16 Python
使用css实现android系统的loading加载动画
2019/07/25 HTML / CSS
HTML5 LocalStorage 本地存储刷新值还在
2017/03/10 HTML / CSS
卡骆驰新加坡官网:Crocs新加坡
2018/06/12 全球购物
俄罗斯厨房产品购物网站:COOK HOUSE
2021/03/15 全球购物
J2EE模式面试题
2016/10/11 面试题
物流管理专业毕业生自荐信
2014/03/04 职场文书
巾帼文明岗汇报材料
2014/12/24 职场文书