PHP函数重载


195

来自C ++背景;)
如何重载PHP函数?

一个函数定义(如果有参数),另一个函数定义(无参数)?PHP中有可能吗?还是应该使用if来检查$ _GET和POST是否传递了任何参数?并联系他们?


1
您只能重载类方法,而不能重载函数。参见php.net/manual/en/language.oop5.overloading.php
Spechal,2011年

1
您可以创建一个函数来显式检查参数的数量,并从一组预定义的参数中执行另一个函数。但是,您最好重新设计解决方案,或者使用实现您的接口的类
kolypto 2011年

2
正如php.net/manual/en/language.oop5.overloading.php所说,PHP对重载的定义不同于典型的OOP语言。他们只是指的魔术方法,使动态特性和功能的路由基于X.
埃德温·丹尼尔斯

对于未来的读者:@Spechal所指的overloading是与该问题所要求的含义不同的单词。(有关更多详细信息,请参见接受的答案。)
ToolmakerSteve15年

2
自PHP 7以来发生了什么变化?:o
nawfal

Answers:


217

您不能重载PHP函数。函数签名仅基于它们的名称,并且不包括参数列表,因此不能有两个具有相同名称的函数。PHP中的类方法重载与许多其他语言中的不同。PHP使用相同的词,但是描述了不同的模式。

但是,您可以声明一个可变参量,该可变参函数接受可变数量的参数。您将使用func_num_args()func_get_arg()获取参数,并正常使用它们。

例如:

function myFunc() {
    for ($i = 0; $i < func_num_args(); $i++) {
        printf("Argument %d: %s\n", $i, func_get_arg($i));
    }
}

/*
Argument 0: a
Argument 1: 2
Argument 2: 3.5
*/
myFunc('a', 2, 3.5);

8
也许我做了太多的C ++开发工作,但是我建议您可以在函数参数(如)中完成此操作myFunc(/*...*/)
doug65536 2013年

4
@ doug65536,PHP 5.6+将支持将“ ...”作为语法标记,以减轻我们的负担。;)
Sz。

或参见Adil的answer,它更接近于C ++的重载-使用像php这样的松散类型的语言,尽可能地获得重载。它甚至更适合php 7,因为如果所有重载中的参数类型相同,则可以为参数提供类型提示。
ToolmakerSteve

78

PHP不支持传统的方法重载,但是您可能能够实现所需目标的一种方法是利用__callmagic方法:

class MyClass {
    public function __call($name, $args) {

        switch ($name) {
            case 'funcOne':
                switch (count($args)) {
                    case 1:
                        return call_user_func_array(array($this, 'funcOneWithOneArg'), $args);
                    case 3:
                        return call_user_func_array(array($this, 'funcOneWithThreeArgs'), $args);
                 }
            case 'anotherFunc':
                switch (count($args)) {
                    case 0:
                        return $this->anotherFuncWithNoArgs();
                    case 5:
                        return call_user_func_array(array($this, 'anotherFuncWithMoreArgs'), $args);
                }
        }
    }

    protected function funcOneWithOneArg($a) {

    }

    protected function funcOneWithThreeArgs($a, $b, $c) {

    }

    protected function anotherFuncWithNoArgs() {

    }

    protected function anotherFuncWithMoreArgs($a, $b, $c, $d, $e) {

    }

}

20
我以前没有看过这种用法__call()。很有创意(有点冗长)!+1
BoltClock

真正令人钦佩的使用__call的()
阿布舍克古普塔

2
实际上,不能同意这一点,必须对此建议进行再培训。例如,__ call()的使用是一种反模式。其次,对于可见性正确的类方法,可以在PHP中进行重载。但是,您不能-重载纯jane函数。
奥德曼(Oddman)

1
您能解释一下为什么您认为使用__call()是反模式吗?PHP方法重载不是OP所寻找的-他们希望具有相同名称但输入/输出不同的多个方法签名的能力:en.wikipedia.org/wiki/Function_overloading
Stephen

20
无需使用__call()。而是声明一个具有所需名称的方法,不列出任何参数,然后在该方法内使用func_get_args()调度到适当的私有实现。
FantasticJamieBurns

30

要重载函数,只需在默认情况下将参数传递为null即可,

class ParentClass
{
   function mymethod($arg1 = null, $arg2 = null, $arg3 = null)  
     {  
        if( $arg1 == null && $arg2 == null && $arg3 == null ){ 
           return 'function has got zero parameters <br />';
        }
        else
        {
           $str = '';
           if( $arg1 != null ) 
              $str .= "arg1 = ".$arg1." <br />";

           if( $arg2 != null ) 
              $str .= "arg2 = ".$arg2." <br />";

           if( $arg3 != null ) 
              $str .= "arg3 = ".$arg3." <br />";

           return $str;
         }
     }
}

// and call it in order given below ...

 $obj = new ParentClass;

 echo '<br />$obj->mymethod()<br />';
 echo $obj->mymethod();

 echo '<br />$obj->mymethod(null,"test") <br />';
 echo $obj->mymethod(null,'test');

 echo '<br /> $obj->mymethod("test","test","test")<br />';
 echo $obj->mymethod('test','test','test');

4
我不认为默认参数是函数重载。函数[或方法]重载与基于传递的参数类型调用不同的实现有关。使用默认参数仅允许您使用较少的参数来调用相同的实现。
2013年

1
是的,您也可以基于类型来操作它,但是就好像您知道php松散类型的语言一样,处理它需要解决该问题。
Adil Abbasi 2013年

1
我更喜欢这个答案,而不是公认的答案,因为它明确了参数的最小和最大数量。(不提供必需参数的默认值。)@Scalable-我同意Adil的观点,由于php是松散类型的,因此这实际上是php对overload函数的全部含义-没错,您提出了一个有用的观点读者应该注意的。
制造商史蒂夫(Steve)2015年

11

对于某些人来说这可能有点骇人听闻,但我从Cakephp如何完成某些功能并对其进行了调整中了解了这种方式,因为我喜欢它创建的灵活性

这个想法是您有不同类型的参数,数组,对象等,然后您检测到传递的内容并从那里去

function($arg1, $lastname) {
    if(is_array($arg1)){
        $lastname = $arg1['lastname'];
        $firstname = $arg1['firstname'];
    } else {
        $firstname = $arg1;
    }
    ...
}

1
不,我不认为这是骇人听闻的,PHP的许多内置功能都是这样做的。
BoltClock

因为php是松散类型的,所以这正是人们必须处理这种情况的方式。它在php中的“必要的技巧”。
ToolmakerSteve

11
<?php   
/*******************************
 * author  : hishamdalal@gmail.com 
 * version : 3.8
 * create on : 2017-09-17
 * updated on : 2020-01-12
 *****************************/

#> 1. Include Overloadable class

class Overloadable
{
    static function call($obj, $method, $params=null) {
        $class = get_class($obj);
        // Get real method name
        $suffix_method_name = $method.self::getMethodSuffix($method, $params);

        if (method_exists($obj, $suffix_method_name)) {
            // Call method
            return call_user_func_array(array($obj, $suffix_method_name), $params);
        }else{
            throw new Exception('Tried to call unknown method '.$class.'::'.$suffix_method_name);
        }
    }

    static function getMethodSuffix($method, $params_ary=array()) {
        $c = '__';
        if(is_array($params_ary)){
            foreach($params_ary as $i=>$param){
                // Adding special characters to the end of method name 
                switch(gettype($param)){
                    case 'array':       $c .= 'a'; break;
                    case 'boolean':     $c .= 'b'; break;
                    case 'double':      $c .= 'd'; break;
                    case 'integer':     $c .= 'i'; break;
                    case 'NULL':        $c .= 'n'; break;
                    case 'object':
                        // Support closure parameter
                        if($param instanceof Closure ){
                            $c .= 'c';
                        }else{
                            $c .= 'o'; 
                        }
                    break;
                    case 'resource':    $c .= 'r'; break;
                    case 'string':      $c .= 's'; break;
                    case 'unknown type':$c .= 'u'; break;
                }
            }
        }
        return $c;
    }
    // Get a reference variable by name
    static function &refAccess($var_name) {
        $r =& $GLOBALS["$var_name"]; 
        return $r;
    }
}
//----------------------------------------------------------
#> 2. create new class
//----------------------------------------------------------

class test 
{
    private $name = 'test-1';

    #> 3. Add __call 'magic method' to your class

    // Call Overloadable class 
    // you must copy this method in your class to activate overloading
    function __call($method, $args) {
        return Overloadable::call($this, $method, $args);
    }

    #> 4. Add your methods with __ and arg type as one letter ie:(__i, __s, __is) and so on.
    #> methodname__i = methodname($integer)
    #> methodname__s = methodname($string)
    #> methodname__is = methodname($integer, $string)

    // func(void)
    function func__() {
        pre('func(void)', __function__);
    }
    // func(integer)
    function func__i($int) {
        pre('func(integer '.$int.')', __function__);
    }
    // func(string)
    function func__s($string) {
        pre('func(string '.$string.')', __function__);
    }    
    // func(string, object)
    function func__so($string, $object) {
        pre('func(string '.$string.', '.print_r($object, 1).')', __function__);
        //pre($object, 'Object: ');
    }
    // func(closure)
    function func__c(Closure $callback) {

        pre("func(".
            print_r(
                array( $callback, $callback($this->name) ), 
                1
            ).");", __function__.'(Closure)'
        );

    }   
    // anotherFunction(array)
    function anotherFunction__a($array) {
        pre('anotherFunction('.print_r($array, 1).')', __function__);
        $array[0]++;        // change the reference value
        $array['val']++;    // change the reference value
    }
    // anotherFunction(string)
    function anotherFunction__s($key) {
        pre('anotherFunction(string '.$key.')', __function__);
        // Get a reference
        $a2 =& Overloadable::refAccess($key); // $a2 =& $GLOBALS['val'];
        $a2 *= 3;   // change the reference value
    }

}

//----------------------------------------------------------
// Some data to work with:
$val  = 10;
class obj {
    private $x=10;
}

//----------------------------------------------------------
#> 5. create your object

// Start
$t = new test;

#> 6. Call your method

// Call first method with no args:
$t->func(); 
// Output: func(void)

$t->func($val);
// Output: func(integer 10)

$t->func("hello");
// Output: func(string hello)

$t->func("str", new obj());
/* Output: 
func(string str, obj Object
(
    [x:obj:private] => 10
)
)
*/

// call method with closure function
$t->func(function($n){
    return strtoupper($n);
});

/* Output:
func(Array
(
    [0] => Closure Object
        (
            [parameter] => Array
                (
                    [$n] => 
                )

        )

    [1] => TEST-1
)
);
*/

## Passing by Reference:

echo '<br><br>$val='.$val;
// Output: $val=10

$t->anotherFunction(array(&$val, 'val'=>&$val));
/* Output:
anotherFunction(Array
(
    [0] => 10
    [val] => 10
)
)
*/

echo 'Result: $val='.$val;
// Output: $val=12

$t->anotherFunction('val');
// Output: anotherFunction(string val)

echo 'Result: $val='.$val;
// Output: $val=36







// Helper function
//----------------------------------------------------------
function pre($mixed, $title=null){
    $output = "<fieldset>";
    $output .= $title ? "<legend><h2>$title</h2></legend>" : "";
    $output .= '<pre>'. print_r($mixed, 1). '</pre>';
    $output .= "</fieldset>";
    echo $output;
}
//----------------------------------------------------------

4
您能否添加一些有关如何使用此类的说明?
Justus Romijn

1-创建新的类2-扩展可重载。3-创建函数,例如funcname_()=>无参数,或像funcname_s($ s)=> string arg </ li>
Hisham Dalal 2013年

1
这是很酷的解决方案。为什么要做$ o = new $ obj()?我还没有尝试过,尽管我认为应该是\ $ o = \ $ this?
13年

感谢您的这一重要通知,我将使用反斜杠,但无论有没有反斜杠,它都可以使用!-我将phpEazy用作本地服务器。
Hisham Dalal


3

在PHP 5.6中,您可以将splat运算符 ...用作最后一个参数,并取消使用func_get_args()and func_num_args()

function example(...$args)
{
   count($args); // Equivalent to func_num_args()
}

example(1, 2);
example(1, 2, 3, 4, 5, 6, 7);

您也可以使用它来解压缩参数:

$args[] = 1;
$args[] = 2;
$args[] = 3;
example(...$args);

等效于:

example(1, 2, 3);

1
<?php

    class abs
    {
        public function volume($arg1=null, $arg2=null, $arg3=null)
        {   
            if($arg1 == null && $arg2 == null && $arg3 == null)
        {
            echo "function has no arguments. <br>";
        }

        else if($arg1 != null && $arg2 != null && $arg3 != null)
            {
            $volume=$arg1*$arg2*$arg3;
            echo "volume of a cuboid ".$volume ."<br>";
            }
            else if($arg1 != null && $arg2 != null)
            {
            $area=$arg1*$arg2;
            echo "area of square  = " .$area ."<br>";
            }
            else if($arg1 != null)
            {
            $volume=$arg1*$arg1*$arg1; 
            echo "volume of a cube = ".$volume ."<br>";
            }


        }


    }

    $obj=new abs();
    echo "For no arguments. <br>";
    $obj->volume();
    echo "For one arguments. <br>";
    $obj->volume(3);
    echo "For two arguments. <br>";
    $obj->volume(3,4);
    echo "For three arguments. <br>";
    $obj->volume(3,4,5);
    ?>

尝试编辑问题并使用格式。这将使您的答案更具可读性,并吸引更多用户。
Kashish Arora



0

可悲的是,PHP中没有C#中的重载。但是我有一个小把戏。我用默认的空值声明参数并在函数中检查它们。这样,我的函数可以根据参数执行不同的操作。下面是简单的示例:

public function query($queryString, $class = null) //second arg. is optional
{
    $query = $this->dbLink->prepare($queryString);
    $query->execute();

    //if there is second argument method does different thing
    if (!is_null($class)) { 
        $query->setFetchMode(PDO::FETCH_CLASS, $class);
    }

    return $query->fetchAll();
}

//This loads rows in to array of class
$Result = $this->query($queryString, "SomeClass");
//This loads rows as standard arrays
$Result = $this->query($queryString);

1
请写完所有新答案后再写一年。上面的答案中已经两次显示了此技术。一次是在2013年,一次是在2014
。– ToolmakerSteve19年
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.