PHP反射机制原理与用法

反射

反射,直观理解就是根据到达地找到出发地和来源。比如,一个光秃秃的对象,我们可以仅仅通过这个对象就能知道它所属的类、拥有哪些方法

反射是指在PHP运行状态中,扩展分析PHP程序,导出或提出关于类、方法、属性、参数等的详细信息,包括注释。这种动态获取信息以及动态调用对象方法的功能称为反射API。

 

php类的成员变量没有在类中声明,而是在函数中声明,有什么不同?

    ReflectionClass: 获取类声明时的结构

    ReflectionObject: 可获取类实例化后的结构

 

代码如下:
class test{
    private $name;
    private $sex;
    function __construct(){
        $this->aaa='aaa';
    }
}
$test=new test();
$reflect=new ReflectionClass($test);
$pro=$reflect->getDefaultProperties();
print_r($pro);//打印结果:Array ( [name] => [sex] => )
echo $test->aaa;//打印结果:aaa


在这个test类中,声明了两个成员变量$name和$sex,但是在构造函数中,又声明了一个变量$aaa,初始化类,使用反射类打印默认成员属性只有声明的两个成员变量属性,但是打印类的$aaa变量发现还是可以输出结果。
请问类的成员变量不用声明,在函数中声明也是可以的吗,有什么区别?
在你这个例子中,使用ReflectionClass是不恰当的,因为__construct只有在实例化class时,才会执行。
也就是说ReflectionClass更多的是反射类声明时的结构,而不是类实例化后的结构,所以没有输出属性aaa是正确,因为属性aaa确实是(在类声明时)不存在的。
那么怎么看属性aaa呢,应该用ReflectionObject反射实例化后的结构,例如

代码如下:
<?php
class test{
    private $name;
    private $sex;
    function __construct(){
        $this->aaa='aaa';
    }
}
$test=new test();
$reflect=new ReflectionObject($test);
$pro=$reflect->getProperties();
print_r($pro);


经过实例化以后,属性aaa才会存在,这时你就能看到属性aaa了

 

如何使用反射API

例子:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

<?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 '未设置';

       

     $this->$name="正在为你设置默认值";

  }

  return $this->$name;

  }

}

$student=new person();

$student->name='Tom';

$student->gender='male';

$student->age=24;

现在,要获取这个student对象的方法和属性列表该怎么做呢?如以下代码所示:

1

2

3

4

5

6

7

8

9

10

11

// 获取对象属性列表

$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,使用class函数,返回对象属性的关联数组以及更多的信息:

1

2

3

4

5

6

// 返回对象属性的关联数组

var_dump(get_object_vars($student));

// 类属性

var_dump(get_class_vars(get_class($student)));

// 返回由类的方法名组成的数组

var_dump(get_class_methods(get_class($student)));

假如这个对象是从其他页面传过来的,怎么知道它属于哪个类呢?一句代码就可以搞定:

1

2

// 获取对象属性列表所属的类

echo get_class($student);

反射API的功能显然更强大,甚至能还原这个类的原型,包括方法的访问权限等,如:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

// 反射获取类的原型

$obj = new ReflectionClass('person');

$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() ? ' public' : '',$v->isPrivate() ? ' private' : '',

    $v->isProtected() ? ' protected' : '',

    $v->isStatic() ? ' static' : '';

    echo "\t{$k}\n";

}

echo "\n";

if(is_array($Methods)) ksort($Methods);

foreach($Methods as $k => $v){

    echo "\tfunction {$k}(){}\n";

}

echo "}\n";

输出如下:

1

2

3

4

5

6

7

8

class person

{

  public gender

  public name

  function get(){}

  function set(){}

  function say(){}

}

不仅如此,PHP手册中关于反射API更是有几十个,可以说,反射完整地描述了一个类或者对象的原型。反射不仅可以用于类和对象,还可以用于函数、扩展模块、异常等。

 

反射有什么作用

反射可以用于文档生成。因此可以用它对文件里的类进行扫描,逐个生成描述文档。

既然反射可以探知类的内部结构,那么是不是可以用它做hook实现插件功能呢?或者是做动态代理呢?

例如:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

<?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('mysql');

$obj->connect('member');

在平常开发中,用到反射的地方不多:一个是对对象进行调试,另一个是获取类的信息。在MVC和插件开发中,使用反射很常见,但是反射的消耗也很大,在可以找到替代方案的情况下,就不要滥用。

PHP有Token函数,可以通过这个机制实现一些反射功能。从简单灵活的角度讲,使用已经提供的反射API是可取的。

很多时候,善用反射能保持代码的优雅和简洁,但反射也会破坏类的封装性,因为反射可以使本不应该暴露的方法或属性被强制暴露了出来,这既是优点也是缺点。

转载于:https://my.oschina.net/ganfanghua/blog/1606326

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值