php后期静态绑定例子讲解

本文介绍了PHP从5.3.0版本开始引入的后期静态绑定特性,用于在继承范围内引用静态调用的类。文章通过示例展示了self::、static::的区别,以及它们在非静态环境下的行为。此外,还探讨了后期静态绑定如何解析到完全确定的静态调用,并解释了parent::和self::的转发调用机制。
摘要由CSDN通过智能技术生成

自 PHP 5.3.0 起,PHP 增加了一个叫做后期静态绑定的功能,用于在继承范围内引用静态调用的类。 

1.self::限制

使用 self:: 或者 __CLASS__ 对当前类的静态引用,取决于定义当前方法所在的类:

<?php
class A {
    public static function who() {
        echo __CLASS__;
    }
    public static function test() {
        self::who();
    }
}

class B extends A {
    public static function who() {
        echo __CLASS__;
    }
}

B::test();
?> 
解释:定义当前test方法的类为A,所以self代表类A。

输出结果为:A

static:: 

<?php
class A {
    public static function who() {
        echo __CLASS__;
    }
    public static function test() {
        static::who(); // 后期静态绑定从这里开始
    }
}

class B extends A {
    public static function who() {
        echo __CLASS__;
    }
}

B::test();
?> 
解释:static::不再被解析为定义当前方法所在的类,而是在实际运行时计算的。

输出结果为:B


2.在非静态环境下,所调用的类即为该对象实例所属的类。由于 $this-> 会在同一作用范围内尝试调用私有方法,而static:: 则可能给出不同结果。另一个区别是static:: 只能用于静态属性。 

class A {
    private function foo() {
        echo "success!\n";
    }
    public function test() {
        $this->foo();
        static::foo();
    }
}

class B extends A {
   /* foo() will be copied to B, hence its scope will still be A and
    * the call be successful */
}

class C extends A {
    private function foo() {
        /* original method is replaced; the scope of the new one is C */
    }
}

$b = new B();
$b->test();
$c = new C();
$c->test();
解释:类C中没有test()方法,所以类C的实例调用父类A中的test方法,此时处于父类A的作用于域中,无法调用C::foo()这私有方法,所以在使用static::foo()时会报错 Fatal error: Call to private method C::foo() from context 'A' ,但$this会尝试调用同一作用域中的私有方法,即无法调用当前对象c的方法,就调用它父类的。

输出结果为:

success! success! success! 

( ! ) Fatal error: Call to private method C::foo() from context 'A' 

3.后期静态绑定的解析会一直到取得一个完全解析了的静态调用为止。另一方面,如果静态调用使用 parent:: 或者self:: 将转发调用信息。

 转发和非转发调用

<?php
class A {
    public static function foo() {
        static::who();
    }

    public static function who() {
        echo __CLASS__."\n";
    }
}

class B extends A {
    public static function test() {
        A::foo();
        parent::foo();
        self::foo();
    }

    public static function who() {
        echo __CLASS__."\n";
    }
}
class C extends B {
    public static function who() {
        echo __CLASS__."\n";
    }
}

C::test();
?> 
解释:

所谓的"转发调用"(forwarding call)指的是通过以下几种方式进行的静态调用:self::parent::static:: 以及forward_static_call()。可用get_called_class() 函数来得到被调用的方法所在的类名,static:: 则指出了其范围。 

A::foo() 静态直接指名到姓的调用A内静态函数,输出A

 parent::foo()是调用上一级的父类中的方法 ,此处为A,self::调用自身(类B)的foo()方法,类B中没有foo()方法,则调用类A中的foo()方法。

A,B,C三个类里都有同一个名称who()方法,根据覆盖效应系统会用优先级最高的,即C中的。输出:C C


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值