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生成随机数或者字符串的代码
Sep 05 PHP
PHP 读取和修改大文件的某行内容的代码
Oct 30 PHP
PHP错误Cannot use object of type stdClass as array in错误的解决办法
Jun 12 PHP
20个2014年最优秀的PHP框架回顾
Oct 22 PHP
PHP实现删除字符串中任何字符的函数
Aug 11 PHP
ThinkPHP数据操作方法总结
Sep 28 PHP
PHP实现二维数组按某列进行排序的方法
Nov 18 PHP
PHP树形结构tree类用法示例
Feb 01 PHP
PHP快速排序算法实现的原理及代码详解
Apr 03 PHP
详解PHP素材图片上传、下载功能
Apr 12 PHP
Laravel Eloquent ORM 多条件查询的例子
Oct 10 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
php+dbfile开发小型留言本
2006/10/09 PHP
火车头discuz6.1 完美采集的php接口文件
2009/09/13 PHP
php中比较简单的导入phpmyadmin生成的sql文件的方法
2011/06/28 PHP
PHP 闭包详解及实例代码
2016/09/28 PHP
JavaScript下申明对象的几种方法小结
2008/10/02 Javascript
js bind 函数 使用闭包保存执行上下文
2011/12/26 Javascript
JavaScript日期时间与时间戳的转换函数分享
2015/01/31 Javascript
Vue 进阶之路(三)
2019/04/18 Javascript
vue+Element实现搜索关键字高亮功能
2019/05/28 Javascript
Vue中axios拦截器如何单独配置token
2019/12/27 Javascript
VUE使用axios调用后台API接口的方法
2020/08/03 Javascript
vue 使用post/get 下载导出文件操作
2020/08/07 Javascript
[50:54]完美世界DOTA2联赛 GXR vs IO 第三场 11.07
2020/11/10 DOTA
详解设计模式中的工厂方法模式在Python程序中的运用
2016/03/02 Python
基础的十进制按位运算总结与在Python中的计算示例
2016/06/28 Python
python查看微信好友是否删除自己
2016/12/19 Python
python编程实现希尔排序
2017/04/13 Python
Python中的is和==比较两个对象的两种方法
2017/09/06 Python
python编程嵌套函数实例代码
2018/02/11 Python
python2.7读取文件夹下所有文件名称及内容的方法
2018/02/24 Python
python微元法计算函数曲线长度的方法
2018/11/08 Python
python使用pipeline批量读写redis的方法
2019/02/18 Python
浅谈python之高阶函数和匿名函数
2019/03/21 Python
Python转换itertools.chain对象为数组的方法
2020/02/07 Python
Pandas DataFrame求差集的示例代码
2020/12/13 Python
html5声频audio和视频video等新特性详细说明
2012/12/26 HTML / CSS
Under Armour安德玛中国官网:美国高端运动科技品牌
2018/03/09 全球购物
英国莱斯特松木橡木家具网上商店:Choice Furniture Superstore
2019/07/05 全球购物
.NET里面什么时候需要调用垃圾回收
2015/06/01 面试题
电气专业推荐信范文
2013/11/18 职场文书
初中生评语大全
2014/04/24 职场文书
小学生思想品德评语
2014/12/31 职场文书
出纳岗位职责
2015/01/31 职场文书
导游词之重庆钓鱼城
2019/09/19 职场文书
MySQL之PXC集群搭建的方法步骤
2021/05/25 MySQL
Java 超详细讲解设计模式之中的抽象工厂模式
2022/03/25 Java/Android