PHP 5.3.0부터 PHP에는 상속 범위에서 정적으로 호출된 클래스를 참조하는 데 사용되는 후기 정적 바인딩이라는 기능이 추가되었습니다.
정확히 말하면 후기 정적 바인딩의 작동 원리는 이전 "비전달 호출"에 클래스 이름을 저장하는 것입니다. 정적 메서드 호출을 수행할 때 클래스 이름은 명시적으로 지정된 이름입니다(일반적으로 :: 연산자의 왼쪽에 있음). 비정적 메서드 호출을 수행할 때는 개체가 속한 클래스입니다. 소위 "전달 호출"은 self::, parent::, static:: 및 전달_정적_call() 방식으로 이루어진 정적 호출을 의미합니다. get_called_class() 함수를 사용하여 호출된 메서드의 클래스 이름을 가져올 수 있으며 static::은 해당 범위를 가리킵니다.
이 기능은 언어 내부 관점에서 "Late Static Binding"이라고 불립니다. "후기 바인딩"은 static::이 더 이상 현재 메서드가 정의된 클래스로 구문 분석되지 않고 실제 런타임에 계산된다는 것을 의미합니다. 정적 메서드 호출에 사용할 수 있으므로(그러나 이에 국한되지는 않음) "정적 바인딩"이라고도 합니다.
self::
의 제한 사항 현재 메서드가 정의된 클래스에 따라 현재 클래스에 대한 정적 참조에 self:: 또는 __CLASS__를 사용합니다.
예제 #1 self: : Usage
<?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(); ?>
위 루틴은 다음을 출력합니다:
A
Usage of late static Binding
Late Static Binding은 원래 new 키워드는 런타임이 제한을 우회하기 위해 처음에 호출하는 클래스를 나타냅니다. 간단히 말해서, 이 키워드를 사용하면 위 예제에서 test()를 호출할 때 A 클래스 대신 B 클래스를 참조할 수 있습니다. 결국 새로운 키워드를 도입하지 않고 이미 예약된 정적 키워드를 사용하기로 결정되었습니다.
예 #2 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(); ?>
위 루틴은 다음을 출력합니다.
B
참고:
static 환경에서 호출되는 클래스는 객체 인스턴스가 속한 클래스입니다. $this->는 동일한 범위에서 개인 메소드를 호출하려고 시도하므로 static::은 다른 결과를 제공할 수 있습니다. 또 다른 차이점은 static::은 정적 속성에만 사용할 수 있다는 것입니다.
예제 #3 static 사용:
<?php 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(); //fails ?>
위 루틴은 다음을 출력합니다.
success! success! success! Fatal error: Call to private method C::foo() from context 'A' in /tmp/test.php on line 9
참고:
나중에 정적 바인딩 해결이 계속됩니다. 완전히 해결된 정적 호출을 얻을 때까지. 반면에 parent:: 또는 self::를 사용하여 정적 호출이 이루어지면 호출 정보가 전달됩니다.
예 #4 착신 전환 및 비착신 통화
<?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(); ?>
위 루틴은 다음을 출력합니다.
A C C