>  기사  >  백엔드 개발  >  PHP 리플렉션 API 간단한 튜토리얼

PHP 리플렉션 API 간단한 튜토리얼

伊谢尔伦
伊谢尔伦원래의
2017-07-01 10:40:301092검색

리플렉션 API에 대해 말하자면, PHP의 리플렉션 API는 Java의 java.lang.reflect 패키지와 비슷하다고 생각합니다. 둘 다 클래스 멤버 속성과 메서드를 인쇄하고 분석할 수 있는 내장 클래스 집합으로 구성되어 있습니다. . 어쩌면 get_class_vars()와 같은 객체 함수를 이미 배웠을 수도 있지만 리플렉션 API를 사용하는 것이 더 유연하고 출력 정보가 ​​더 자세할 것입니다.

먼저 리플렉션 API는 클래스를 확인하는 데만 사용되는 것이 아니라 자체적으로 다양한 기능을 완성하는 클래스 집합을 포함한다는 점을 알아야 합니다. 일반적으로 사용되는 클래스는 다음과 같습니다.

Reflection class 클래스의 기본 정보를 출력할 수 있습니다(제공된 static import() 함수를 통해)
ReflectionMethod 클래스 이름을 보고 의미를 알고, 클래스 메소드를 출력하세요 , 메소드 등에 대한 구체적인 정보를 가져옵니다.
ReflectionClass 클래스 는 클래스에 포함된 메소드, 클래스의 속성, 추상 여부 등 클래스 정보를 가져오는 데 사용됩니다. 클래스 등
ReflectionParameter 클래스 동적일 수 있는 매개변수 정보 표시 알려진 메소드의 매개변수 전달 상황을 가져옵니다.
ReflectionException 클래스 오류 메시지를 표시하는 데 사용됩니다
ReflectionExtension 클래스 PHP 확장 정보를 얻으면 판단할 수 있습니다. 잠깐
ㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋㅋ


기존 인쇄정보와 리플렉션 API의 차이점
다음은 리플렉션 사용을 보여주기 위해 제가 직접 작성한 매개변수 프로그램입니다.

<?php

class Person
{
    //成员属性
    public $name;
    public $age; 

    //构造方法
    public function construct($name, $age)
    {
        $this->name = $name;
        $this->age = $age;
    }

    //成员方法
    public function set_name($name)
    {
        $this->$name = $name;
    }

    public function get_name()
    {
        return $this->$name;
    }

    public function get_age()
    {
        return $this->$age;
    }

    public function get_user_info()
    {
        $info = &#39;姓名:&#39; . $this->name;
        $info .= &#39; 年龄:&#39; . $this->age;
        return $info;
    }
}

class Teacher extends Person
{
    private $salary = 0;

    public function construct($name, $age, $salary)
    {
        parent::construct($name, $age);
        $this->salary = $salary;
    }

    public function get_salary()
    {
        return $this->$salary;
    }

    public function get_user_info()
    {
        $info = parent::get_user_info();
        $info .= " 工资:" . $this->salary;
        return $info;
    }
}

class Student extends Person
{
    private $score = 0;

    public function construct($name, $age, $score)
    {
        parent::construct($name, $age);
        $this->score = $score;
    }

    public function get_score()
    {
        return $this->score;        
    }

    public function get_user_info()
    {
        $info = parent::get_user_info();
        $info .= " 成绩:" . $this->score;
        return $info;
    }
}

header("Content-type:text/html;charset=utf8;");
$te_obj = new Teacher(&#39;李老师&#39;, &#39;36&#39;, &#39;2000&#39;);
$te_info = $te_obj->get_user_info();

$st_obj = new Student(&#39;小明&#39;, &#39;13&#39;, &#39;80&#39;);
$st_info = $st_obj->get_user_info();

먼저 var_dump()를 사용하여 클래스 정보를 인쇄합니다. 아래와 같이 그냥 인쇄되는 것을 볼 수 있습니다. 수업에 대한 간단한 정보는 물론 메소드조차도 존재하지 않으므로 이러한 정보에서는 다른 수영정보를 알 수 없습니다. Reflection에서 제공하는 내장 내보내기 방법을 사용하여 아래와 같이 정보를 인쇄합니다. 인쇄된 정보는 멤버 속성, 멤버 메서드, 클래스 기본 정보, 파일 경로, 메서드 정보, 메서드 속성, 매개 변수 전송을 포함하여 비교적 완전합니다. 상태, 파일의 줄 수 등 클래스 정보를 비교적 포괄적으로 표시합니다. var_dump()나 print_r은 클래스에 대한 간략한 정보만 표시할 수 있고, 많은 정보는 전혀 표시할 수 없어 간단한 디버깅에만 사용할 수 있음을 알 수 있는데, Reflection API는 클래스에 대한 자세한 정보를 제공합니다. 이는 인터페이스 작성, 특히 다른 사람의 인터페이스 호출에 큰 편의를 제공하는 클래스 호출의 상황을 알고 있습니다. 문제가 발생하는 경우 디버깅에도 도움이 될 수 있습니다.

object(Teacher)#1 (3) {
      ["salary":"Teacher":private]=>
          string(4) "2000"
      ["name"]=>
          string(9) "李老师"
      ["age"]=>
          string(2) "36"
}

Reflection API의 구체적인 사용:


프레임워크의 소스 코드를 읽은 학생들은 프레임워크가 타사 플러그인, 클래스 라이브러리 등을 로드할 수 있다는 것을 알고 있습니다. 다음 예제에서는 이 함수를 간단히 구현하기 위해 Reflection API를 사용했는데, 이 예제의 프로토타입을 책에서 배운 후, 나만의 아이디어에 따라 세트를 작성했습니다. Property 클래스를 탐색하고 호출합니다. 개체와 클래스는 클래스를 기존 코드에 포함하거나 클래스 라이브러리 코드를 수동으로 호출할 필요 없이 다른 클래스의 메서드를 자유롭게 로드할 수 있습니다.
규칙: 각 클래스는 작업 메서드를 포함해야 하며 인터페이스를 추상화할 수 있습니다. 각 클래스 라이브러리의 정보에 해당하는 각 클래스의 정보를 파일에 넣은 후, 해당 클래스가 저장한 Property 클래스 라이브러리의 해당 객체를 통해 각 클래스 라이브러리의 작업 메소드를 호출할 수 있습니다.

다음은 기본 코드입니다.

object(Teacher)#1 (3) {
      ["salary":"Teacher":private]=>
          string(4) "2000"
      ["name"]=>
          string(9) "李老师"
      ["age"]=>
          string(2) "36"
}
Class [  class Person ] {
      @@ /usr/local/www/phptest/oop/reflaction.php 3-38
      - Constants [0] {
      }
      - Static properties [0] {
      }
      - Static methods [0] {
 }
  - Properties [2] {
        Property [  public $name ]
        Property [  public $age ]
  }

  - Methods [5] {
    Method [  public method construct ] {
      @@ /usr/local/www/phptest/oop/reflaction.php 10 - 14

      - Parameters [2] {
        Parameter #0 [  $name ]

.....

여기에는 Property 인터페이스를 구현하기 위해 두 개의 유사한 클래스가 정의되어 있으며 둘 다 단순히 work() 메서드를 구현합니다. 매개 변수에는 Person 개체가 필요합니다. , 파일을 사용하여 저장할 수도 있으며 각 클래스의 정보를 멤버 속성으로 바꿀 수도 있습니다.

/*属性接口*/
interface Property
{
    function work();
}

class Person
{
    public $name;
    public function construct($name)
    {
        $this->name = $name;
    }
}

class StudentController implements Property
{
    //set方法,但需要Person对象参数
    public function setPerson(Person $obj_person)
    {
        echo &#39;Student &#39; . $obj_person->name;
    }

    //work方法简单实现
    public function work()
    {
        echo &#39;student working!&#39;;
    }
}

class EngineController implements Property
{
    //set方法
    public function setWeight($weight)
    {
        echo &#39;this is engine -> set weight&#39;;
    }

    public function setPrice($price)
    {
        echo "this is engine -> set price";
    }

    //work方法简单实现
    public function work()
    {
        echo &#39;engine working!&#39;;
    }
}
 이 프로그램이 끝나면 Run 시작은 생성자 메서드를 자동으로 호출하여 해당 메서드 작업의 초기화 및 실행을 포함하여 로드할 클래스 라이브러리의 다른 멤버 속성을 초기화합니다. 여기서는 해당 set 메서드만 해당됩니다. 완료되었습니다. $mod_arr 속성은 호출 클래스의 모든 개체를 저장합니다. 각 개체에는 데이터가 포함되어 있으며 포함된 개체를 탐색하여 work() 메서드를 호출할 수 있습니다.

 이 프로그램은 리플렉션 PAI를 이해하는 데에만 사용됩니다. 각 기능은 완벽하지 않습니다. 아래에 각 방법에 대한 요약이 있습니다.

Reflection API에서 제공하는 공통 클래스 및 함수:

아래에 제공되는 함수는 전부는 아니고 일반적으로 사용되는 함수이고 일부 함수는 전혀 사용하지 않는 함수도 있기 때문에 다음과 같이 작성해야 합니다. 거짓말, 생각 그것들을 모두 보려면 온라인으로 검색하면 많이 있습니다. 제공된 방법 세트를 외울 필요는 없으며, 사용할 때 검토할 수 있습니다.

class Run
{
    public static $mod_arr = [];
    public static $config = [
        &#39;StudentController&#39; => [
            &#39;person&#39; => &#39;xiao ming&#39;
        ],
        &#39;EngineController&#39;  => [
            &#39;weight&#39; => &#39;500kg&#39;,
            &#39;price&#39;  => &#39;4000&#39;
        ]
    ];

    //加载初始化
    public function construct()
    {
        $config = self::$config;
        //用于检查是不是实现类
        $property = new ReflectionClass(&#39;Property&#39;);
        foreach ($config as $class_name => $params) {
            $class_reflect = new ReflectionClass($class_name);
            if(!$class_reflect->isSubclassOf($property)) {//用isSubclassOf方法检查是否是这个对象
                echo &#39;this is  error&#39;;
                continue;
            }

            //得到类的信息
            $class_obj = $class_reflect->newInstance();
            $class_method = $class_reflect->getMethods();

            foreach ($class_method as $method_name) {
                $this->handle_method($class_obj, $method_name, $params);
            }
            array_push(self::$mod_arr, $class_obj);
        }
    }

    //处理方法调用
    public function handle_method(Property $class_obj, ReflectionMethod $method_name, $params)
    {
        $m_name = $method_name->getName();
        $args = $method_name->getParameters();

        if(count($args) != 1 || substr($m_name, 0, 3) != &#39;set&#39;) {    
            return false;
        }
        //大小写转换,做容错处理
        $property = strtolower(substr($m_name, 3));
     
        if(!isset($params[$property])) {
            return false;
        }

        $args_class = $args[0]->getClass();
        echo &#39;<pre class="brush:php;toolbar:false">&#39;;
        if(empty($args_class)) {
            $method_name->invoke($class_obj, $params[$property]); //如果得到的类为空证明需要传递基础类型参数
        } else {
            $method_name->invoke($class_obj, $args_class->newInstance($params[$property])); //如果不为空说明需要传递真实对象
        }
    }
}

//程序开始
new Run();

위 내용은 PHP 리플렉션 API 간단한 튜토리얼의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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