Вызов статической функции из экземпляра


Я пытаюсь вызвать статическую магическую функцию (__callStatic) из члена ее дочернего класса. Проблема в том, что вместо этого он переходит к нестатическому __call.

<?php

ini_set("display_errors", true);

class a
{
    function __call($method, $params)
    {
        echo "instance";
    }

    static function __callStatic($method, $params)
    {
        echo "static";
    }
}

class b extends a
{
    function foo()
    {
        echo static::bar();
        // === echo self::bar();
        // === echo a::bar();
        // === echo b::bar();
    }
}

$b = new b();
echo phpversion()."<br />";
$b->foo();

?>

Вывод:

5.3.6
instance

Как я могу сделать так, чтобы он отображался "статичным"?

Author: Lazlo, 2011-07-19

2 answers

Если вы удалите магический метод "__вызов", ваш код вернет "статический".

В соответствии с http://php.net/manual/en/language.oop5.overloading.php "__callStatic() запускается при вызове недоступных методов в статическом контексте".

Я думаю, что в вашем коде происходит следующее:

  1. Вы вызываете статический метод из нестатического контекста.
  2. Вызов метода выполняется в нестатическом контексте, поэтому PHP ищет волшебный метод "__позвони".
  3. PHP запускает магический метод '_вызов', если он существует. Или, если он не существует, он вызовет "_callstatic".

Вот возможное решение:

class a
{
    static function __callStatic($method, $params)
    {
        $methodList =  array('staticMethod1', 'staticMethod2');

        // check if the method name should be called statically
        if (!in_array($method, $methodList)) {
            return false;
        }

        echo "static";

        return true;
    }

    function __call($method, $params)
    {
         $status = self::__callStatic($method, $params);
         if ($status) {
             return;
         }
         echo "instance";
    }

}

class b extends a
{
    function foo()
    {
        echo static::staticMethod1();
    }

    function foo2()
    {
        echo static::bar();
    }
}

$b = new b();
echo phpversion()."<br />";
$b->foo();
$b->foo2();
 6
Author: satrun77, 2011-07-25 23:18:22

В PHP есть зарезервированные слова self и parent для доступа к статическим методам из класса и/или созданного объекта. parent относится к унаследованным методам от родительского класса.

class b extends a
{
    function foo()
    {
        echo parent::bar();
    }
}

РЕДАКТИРОВАТЬ: Хм, это не помогает... (с использованием PHP 5.3.5)

$b = new b();
$b->foo();  // displays: instance
a::bar();   // displays: static

2-я ПРАВКА: Ха, это работает только в том случае, если вы опустите метод __call()-в классе a.

class a
{
    static function __callStatic($method, $params)
    {
        echo "static";
    }

//  function __call($method, $params)
//    {
//        echo "instance";
//    }
}

class b extends a
{
    function foo()
    {
        echo parent::bar();
    }
}

$b = new b();
$b->foo();  // displays: static
a::bar();   // displays: static
 0
Author: feeela, 2011-07-19 16:01:43