PHP中实现函数重载

zhaowj00 2020-04-08

由于PHP是弱类型语言,因此函数的输入参数类型无法确定(可以使用类型暗示,但是类型暗示无法用在诸如整型,字符串之类的标量类型上),并且对于一个函数,比如只定义了3个输入参数,PHP却运行调用的时候输入4个或者更多的参数。因此基于这2点,注定了PHP中无法重载函数,(类似Javascript语言),也无法有构造函数的重载。

由于实现函数的重载对提高开发效率很有帮助,如果能象C#或者C++那样,那就非常好了。事实上,PHP的提供了一个魔术方法,mixed __call ( string name, array arguments )。这个方法在php手册中也有提及,根据官方文档,称此方法可以实现函数重载。当调用对象中一个不存在的方法的时候,如果定义了__call()方法,则会调用该方法。比如下面的代码:

<?php
class A
{
   function __call ( $name, $arguments )
    {
        echo "__call调用<br/>";
        echo ‘$name为‘.$name."<br/>";
        print_r ($arguments);
    }
}
(new A)->test("test","argument");
?>

运行结果为:

__call调用

$name为test

Array ( [0] => test [1] => argument )

因此只需要利用该魔术方法既可以实现函数重载。

代码如下:

<?php
class A
{
     function __call ($name, $args )
    {
        if($name==‘f‘)
        {
            $i=count($args);
            if (method_exists($this,$f=‘f‘.$i)) {
                call_user_func_array(array($this,$f),$args);
            }
        }
    }
    function f1($a1)
    {
        echo "1个参数".$a1."<br/>";
    }
    function f2($a1,$a2)
    {
        echo "2个参数".$a1.",".$a2."<br/>";
    }
    function f3($a1,$a2,$a3)
    {
          echo "3个参数".$a1.",".$a2.",".$a3."<br/>";
    }
}
(new A)->f(‘a‘);
(new A)->f(‘a‘,‘b‘);
(new A)->f(‘a‘,‘b‘,‘c‘);
?>

输出如下:

1个参数a  
2个参数a,b    
3个参数a,b,c

同样的在PHP中,实现构造函数的重载,也只能变通的实现。实现的关键要素是获取输入参数,并且根据输入参数判断调用哪个方法。下面是一个示例代码

<?php
class A
{
    function __construct()
    {
        echo "执行构造函数<br/>";
$a = func_get_args(); //获取构造函数中的参数
        $i = count($a);
        if (method_exists($this,$f=‘__construct‘.$i)) {
call_user_func_array(array($this,$f),$a);
        }
    }
    function __construct1($a1)
    {
        echo "1个参数".$a1."<br/>";
    }
    function __construct2($a1,$a2)
    {
        echo "2个参数".$a1.",".$a2."<br/>";
    }
function __construct3($a1,$a2,$a3)
    {
          echo "3个参数".$a1.",".$a2.",".$a3."<br/>";
    }
}
$o = new A(‘a‘);
$o = new A(‘a‘,‘b‘);
$o = new A(‘a‘,‘b‘,‘c‘);
?>

执行构造函数  

1个参数a    
执行构造函数    
2个参数a,b    
执行构造函数    
3个参数a,b,c

顺便提一下,和c#等面向对象语言一样,php中,把构造函数设成私有或者protected,就不能实例化该对象了。

转载于:https://blog.51cto.com/cnn237111/1284085

相关推荐