>  기사  >  백엔드 개발  >  PHP 반사 메커니즘의 사용법에 대한 자세한 설명

PHP 반사 메커니즘의 사용법에 대한 자세한 설명

王林
王林원래의
2019-09-02 09:50:384521검색

PHP 반사 메커니즘의 사용법에 대한 자세한 설명

Reflection

객체 지향 프로그래밍에서는 객체에 내성을 갖는 능력이 부여되는데, 이러한 내성은 과정반성입니다.

반성, 직관적 이해는 도착지를 기준으로 출발지와 출처를 찾는 것입니다. 예를 들어, 베어 객체의 경우, 이 객체를 통해서만 그것이 속한 클래스와 어떤 메소드를 가지고 있는지 알 수 있습니다.

Reflection은 PHP 실행 상태에서 PHP 프로그램 분석을 확장하여 주석을 포함한 클래스, 메서드, 속성, 매개 변수 등에 대한 자세한 정보를 내보내거나 제안하는 것을 의미합니다. 동적으로 정보를 얻고 객체 메소드를 동적으로 호출하는 기능을 리플렉션 API라고 합니다.

리플렉션 API는 어떻게 사용하나요?

<?php
class person{
public $name;
public $gender;
public function say(){
  echo $this->name," \tis ",$this->gender,"\r\n";
}
public function set($name, $value) {
  echo "Setting $name to $value \r\n";
  $this->$name= $value;
}
public function get($name) {
  if(!isset($this->$name)){
    echo &#39;未设置&#39;;
       
     $this->$name="正在为你设置默认值";
  }
  return $this->$name;
  }
}
$student=new person();
$student->name=&#39;Tom&#39;;
$student->gender=&#39;male&#39;;
$student->age=24;

이제 이 학생 개체의 메서드와 속성 목록을 얻으려면 어떻게 해야 할까요? 다음 코드에 표시된 대로:

// 获取对象属性列表
$reflect = new ReflectionObject($student);
$props = $reflect->getProperties();
foreach ($props as $prop) {
  print $prop->getName() ."\n";
}
// 获取对象方法列表
$m=$reflect->getMethods();
foreach ($m as $prop) {
  print $prop->getName() ."\n";
}

리플렉션 API 없이 클래스 함수를 사용하여 객체 속성 및 추가 정보의 연관 배열을 반환할 수도 있습니다.

// 返回对象属性的关联数组
var_dump(get_object_vars($student));
// 类属性
var_dump(get_class_vars(get_class($student)));
// 返回由类的方法名组成的数组
var_dump(get_class_methods(get_class($student)));

이 객체가 다른 페이지에서 전달되는데, 어떤 카테고리에 속하는지 어떻게 알 수 있나요? 단 한 줄의 코드로 작업을 완료할 수 있습니다:

// 获取对象属性列表所属的类
echo get_class($student);

리플렉션 API의 기능은 확실히 더 강력하며 메소드의 액세스 권한을 포함하여 이 클래스의 프로토타입을 복원할 수도 있습니다. , 예:

// 反射获取类的原型
$obj = new ReflectionClass(&#39;person&#39;);
$className = $obj->getName();
$Methods = $Properties = array();
foreach($obj->getProperties() as $v)
{
    $Properties[$v->getName()] = $v;
}
foreach($obj->getMethods() as $v)
   
 {
    $Methods[$v->getName()] = $v;
}
echo "class {$className}\n{\n";
is_array($Properties)&&ksort($Properties);
foreach($Properties as $k => $v)
{
    echo "\t";
    echo $v->isPublic() ? &#39; public&#39; : &#39;&#39;,$v->isPrivate() ? &#39; private&#39; : &#39;&#39;,
    $v->isProtected() ? &#39; protected&#39; : &#39;&#39;,
    $v->isStatic() ? &#39; static&#39; : &#39;&#39;;
    echo "\t{$k}\n";
}
echo "\n";
if(is_array($Methods)) ksort($Methods);
foreach($Methods as $k => $v)
{
    echo "\tfunction {$k}(){}\n";
}
echo "}\n";

Output 다음과 같습니다:

class person
{
  public gender
  public name
  function get(){}
  function set(){}
  function say(){}
}

그뿐만 아니라 PHP 매뉴얼에는 수십 개의 리플렉션 API가 있다고 할 수 있습니다. 리플렉션은 클래스나 객체의 프로토타입을 완벽하게 설명합니다. 리플렉션은 클래스와 객체뿐만 아니라 함수, 확장 모듈, 예외 등에 대해서도 사용할 수 있습니다.

반영은 무엇을 하나요?

Reflection은 문서 생성에 사용할 수 있습니다. 따라서 이를 사용하여 파일의 클래스를 스캔하고 설명 문서를 하나씩 생성할 수 있습니다.

리플렉션은 클래스의 내부 구조를 감지할 수 있으므로 플러그인 기능을 구현하기 위한 후크로 사용할 수 있나요? 아니면 동적 프록시인가요?

예:

<?php
class mysql {
  function connect($db) {
    echo "连接到数据库${db[0]}\r\n";
  }
}
class sqlproxy {
  private $target;  
  function construct($tar) { 
    $this->target[] = new $tar();
  }
  function call($name, $args) {
    foreach ($this->target as $obj) {
      $r = new ReflectionClass($obj);
      if ($method = $r->getMethod($name)) {
        if ($method->isPublic() && !$method->isAbstract()) {
          echo "方法前拦截记录LOG\r\n";
          $method->invoke($obj, $args);
          echo "方法后拦截\r\n";
        }
      }
    }
  }
}
$obj = new sqlproxy(&#39;mysql&#39;);
$obj->connect(&#39;member&#39;);

일반적인 개발에서는 리플렉션을 사용하는 곳이 많지 않습니다. 하나는 객체를 디버그하는 것이고, 다른 하나는 클래스 정보를 얻는 것입니다. MVC 및 플러그인 개발에서는 리플렉션을 사용하는 것이 매우 일반적이지만 리플렉션 역시 비용이 많이 듭니다. 대안을 찾을 수 있다면 남용하지 마세요.

많은 경우 리플렉션을 잘 활용하면 코드를 우아하고 간결하게 유지할 수 있지만, 리플렉션은 노출되어서는 안 되는 메서드나 속성을 강제로 노출시킬 수 있으므로 클래스의 캡슐화를 파괴할 수도 있습니다. . 이것은 장점이자 단점입니다.

더 많은 관련 콘텐츠를 보려면 PHP 중국어 웹사이트를 방문하세요: PHP 비디오 튜토리얼

위 내용은 PHP 반사 메커니즘의 사용법에 대한 자세한 설명의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.