ホームページ > php教程 > php手册 > PHP继承方法获取子类名讲解

PHP继承方法获取子类名讲解

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
リリース: 2016-06-13 10:40:54
オリジナル
1069 人が閲覧しました

看了一下datastore的入门,以及开始采用MVC方式来写php,于是想拿php为redis写个model,可以实现一些datastore的基本功能...于是碰到这样一个问题-.-
php里__CLASS__这类东西是静态绑定的,如果不再子类里重载的话,那么继承父类方法所得到的依旧是父类的名称而不是子类的名称。比如:
class A{
        function __construct(){
                echo __CLASS__;
        }
        static function name(){
                echo __CLASS__;
        }
}
class B extends A{}

此时无论将B实例化还是直接调用静态方法,echo出来的都会是A。翻qeephp里是用子类重载的方式解决这个问题,可是这样的话没新搞一个子类就得把相应调用类名的方法重载一边.....这算是php在oop上的缺陷吧,试了试python上没这个问题。
google之。找到两个函数get_class()和get_called_class()。get_class()用于实例调用,加入参数 ($this)可解决子类继承调用的问题,而get_called_class()则是用于静态方法调用,可是...这玩意儿只在php 5.3以后才有....5.3还是比较遥远的事...还好5.2之前可以手动实现这个函数:参阅http://php.net/manual/en /function.get-called-class.php 下方有高手添加了几种5.3之前的实现方式。

if(!function_exists(get_called_class)) {
class class_tools
{
        private static $i = 0;
        private static $fl = null;
        public static function get_called_class()
        {
                $bt = debug_backtrace();
                //使用call_user_func或call_user_func_array函数调用类方法,处理如下
                if (array_key_exists(3, $bt) && array_key_exists(function, $bt[3]) && in_array($bt[3][function], array(call_user_func, call_user_func_array))
) {
                                //如果参数是数组
                                if (is_array($bt[3][args][0])) {
                                $toret = $bt[3][args][0][0];
                                return $toret;
                }else if(is_string($bt[3][args][0])) {//如果参数是字符串
                        //如果是字符串且字符串中包含::符号,则认为是正确的参数类型,计算并返回类名
                        if(false !== strpos($bt[3][args][0], ::)) {
                                $toret = explode(::, $bt[3][args][0]);
                                return $toret[0];
                        }
                }
        }
        //使用正常途径调用类方法,如:A::make()
        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();
        }
}

于是现在可以把例子这么修改:
class A{
        function __construct(){
                echo get_class($this);
        }
        static function name(){
                echo get_called_class();
        }
}
class B extends A{}
这样就能让B直接顺利继承获取当前类名的方法了~

ソース:php.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
最新の問題
PHP 拡張子 intl
から 1970-01-01 08:00:00
0
0
0
phpのデータ取得?
から 1970-01-01 08:00:00
0
0
0
PHP GET エラー レポート
から 1970-01-01 08:00:00
0
0
0
phpを上手に学ぶ方法
から 1970-01-01 08:00:00
0
0
0
人気のおすすめ
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート