如何从扩展PHP类中的静态调用中获取类名?

时间:2009-02-03 11:04:21

标签: php oop inheritance

我有两个课程:ActionMyAction。后者被声明为:

class MyAction extends Action {/* some methods here */}

我需要的只是Action类中的方法(仅在其中,因为会有很多继承的类,我不想在所有类中实现此方法),这将返回静态调用的classname。这就是我在说的:

Class Action {
 function n(){/* something */}
}

当我打电话给他时:

MyAction::n(); // it should return "MyAction"

但是父类中的每个声明只能访问父类__CLASS__变量,该变量的值为“Action”。

有没有办法做到这一点?

7 个答案:

答案 0 :(得分:164)

__CLASS__总是返回使用它的类的名称,因此对静态方法没什么帮助。如果方法不是静态的,您只需使用get_class($ this)。 e.g。

class Action {
    public function n(){
        echo get_class($this);
    }

}

class MyAction extends Action {

}

$foo=new MyAction;

$foo->n(); //displays 'MyAction'

后期静态绑定,可在PHP 5.3 +

中找到

现在发布了PHP 5.3,您可以使用late static bindings,它允许您在运行时解析静态方法调用的目标类,而不是在定义它时。

虽然该功能没有引入一个新的魔术常量来告诉你所调用的类名,但它确实提供了一个新函数get_called_class(),它可以告诉你静态方法被调用的类的名称。这是一个例子:

Class Action {
    public static function n() {
        return get_called_class();
    }
}


class MyAction extends Action {

}


echo MyAction::n(); //displays MyAction

答案 1 :(得分:35)

从5.5开始,你可以use class keyword for the class name resolution,这比进行函数调用快得多。也适用于接口。

// C extends B extends A

static::class  // MyNamespace\ClassC when run in A
self::class    // MyNamespace\ClassA when run in A
parent::class  // MyNamespace\ClassB when run in C
MyClass::class // MyNamespace\MyClass

答案 2 :(得分:16)

这不是理想的解决方案,但适用于PHP< 5.3.0。

代码是从septuro.com

复制而来的
if(!function_exists('get_called_class')) {
    class class_tools {
        static $i = 0;
        static $fl = null;

        static function get_called_class() {
            $bt = debug_backtrace();

            if (self::$fl == $bt[2]['file'].$bt[2]['line']) {
                self::$i++;
            } else {
                self::$i = 0;
                self::$fl = $bt[2]['file'].$bt[2]['line'];
            }

            $lines = file($bt[2]['file']);

            preg_match_all('/([a-zA-Z0-9\_]+)::'.$bt[2]['function'].'/',
                $lines[$bt[2]['line']-1],
                $matches);

            return $matches[1][self::$i];
        }
    }

    function get_called_class() {
        return class_tools::get_called_class();
    }
}

答案 3 :(得分:12)

现在(当5.3到达时)非常简单:

http://php.net/manual/en/function.get-called-class.php

答案 4 :(得分:2)

class MainSingleton { 
  private static $instances = array(); 
  private static function get_called_class() {
    $t = debug_backtrace();
    return $t[count($t)-1]["class"];
  }  

  public static function getInstance() { 
    $class = self::get_called_class();
    if(!isset(self::$instances[$class]) ) { 
      self::$instances[$class] = new $class; 
    } 
    return self::$instances[$class]; 
  } 

}

class Singleton extends MainSingleton { 
  public static function getInstance()
  {
    return parent::getInstance();
  }     
  protected function __construct() { 
    echo "A". PHP_EOL; 
  } 

  protected function __clone() {} 

  public function test() { 
    echo " * test called * "; 
  } 
} 

Singleton::getInstance()->test(); 
Singleton::getInstance()->test();

答案 5 :(得分:0)

在可用的PHP版本中,没有办法做你想做的事。 Paul Dixon的解决方案是唯一的解决方案。我的意思是,代码示例,作为他正在谈论的后期静态绑定功能,从PHP 5.3开始提供,该版本处于测试阶段。

答案 6 :(得分:0)

(PHP 5> = 5.3.0,PHP 7)
get_drawn_class —“最新静态绑定”类名称

<?php

class Model
{
  public static function find()
  {
    return get_called_class();
  }
}

class User extends Model
{
}


echo User::find();

this link might be helpfull