在PHP中,反射是指在PHP运行状态中,扩展分析PHP程序,导出或者提取出关于类、属性、方法、参数等的详细信息,包括注释。这种动态获取信息以及动态调用对象方法的功能,被称为反射API。
其实就是通过一个对象,获取到实例化这个对象的类的所有信息。包括所有属性(共有,私有,静态),所有方法(构造函数,私有函数),还有对应方法的参数信息,还有方法的注释信息等等。
1.反射类对象ReflectionClass
$reflect = new ReflectionClass($class);//$class既可以是类名字符串,也可以是对象。得到一个反射类对象。
$reflect->hasMethod('__make'); //判断类是否有__make方法
$constructor = $reflect->getConstructor();//获取类的构造函数,返回的是ReflectionMethod对象
$obj = $reflect->newInstanceArgs($args);//实例化类,$args是给构造函数的。
//比如最上面的$class如果是一个类名字符串,这个方法相当于 new $class($args);
$reflect->getName();//返回完整的类名
2. ReflectionMethod类,继承ReflectionFunctionAbstract类
2.1如何获得ReflectionMethod的实例
$reflect = new ReflectionClass($class);//$class既可以是类名字符串,也可以是对象。得到一个反射类对象。
//1.获取特定的方法的ReflectionMethod实例
$reflect->getMethod('say'); //say方法的ReflectionMethod实例
$reflect->getConstructor();//获取类的构造函数的ReflectionMethod实例
//2.获取类所有方法的ReflectionMethod实例
$reflect->getMethods(); //返回一个数组,元素就是类所有方法的ReflectionMethod实例
//3.直接传入$class和方法名获得
new ReflectionMethod($class, 'say'); //say方法的ReflectionMethod实例
2.2 ReflectionMethod类的主要功能
$methodObj = new \ReflectionMethod($class, 'say'); //say方法的ReflectionMethod实例
$methodObj->isPublic(); //返回boole值
$methodObj->invokeArgs($class, $arg);//执行say方法,第一参数是调用方法的对象,如果是静态对象,
//设置为 null,第二个参数是传给类方法的参数。
$methodObj->invoke($class);//和上面一样,如果执行的方法是静态类,那么这个参数传送 null。
$methodObj->getNumberOfParameters()//获取say方法的参数个数,返回int
$methodObj->getParameters()//获取say方法的参数,返回数组,元素是ReflectionParameter类的实例
这个$methodObj->invokeArgs($class,$arg);执行这个方法,其中$arg就是这个方法的参数,$arg是一个数组,顺序和方法的形参顺序一致。所以执行这个方法之前,我们还要先得到这个$arg数组。比如tp5.1框架就封装了bindParams来获取这个数组
/**
* 绑定参数
* @access protected
* @param $reflect 就是要执行的那个方法的ReflectionMethod
* @param array $vars 参数
* @return array
*/
protected function bindParams($reflect, $vars = [])
{
//方法没有参数,返回空数组
if ($reflect->getNumberOfParameters() == 0) {
return [];
}
//得到所有参数的
$params = $reflect->getParameters();
foreach ($params as $param) {
$name = $param->getName();
$lowerName = Loader::parseName($name);
$class = $param->getClass();//如果这个参数是某个类实例,返回类名
if ($class) {
$args[] = $this->getObjectParam($class->getName(), $vars);
} elseif (1 == $type && !empty($vars)) {
$args[] = array_shift($vars);
} elseif (0 == $type && isset($vars[$name])) {
$args[] = $vars[$name];
} elseif (0 == $type && isset($vars[$lowerName])) {
$args[] = $vars[$lowerName];
} elseif ($param->isDefaultValueAvailable()) {
$args[] = $param->getDefaultValue();
} else {
throw new InvalidArgumentException('method param miss:' . $name);
}
}
return $args;
}
3.ReflectionParameter 类
$methodObj = new \ReflectionMethod($class, 'say'); //say方法的ReflectionMethod实例
$params = $methodObj->getParameters();//得到数组,元素是ReflectionParameter实例
$paramObj = $params[0];
ReflectionParameter实例的作用
$paramObj->getName();//参数名
$paramObj->getClass();//返回ReflectionClass对象,比如参数标注了传哪个类的实例,这个方法就可以获取到对应的反射类对象。
//所以我们要注入对象的时候,参数要标注完整的类名。
上面三个反射类对应了一个实例的类,方法,方法的参数。实现依赖注入的时候,就是通过反射,找到一个类所要依赖哪些类。