gpt4 book ai didi

PHP通过反射动态加载第三方类和获得类源码的实例

转载 作者:qq735679552 更新时间:2022-09-28 22:32:09 24 4
gpt4 key购买 nike

CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.

这篇CFSDN的博客文章PHP通过反射动态加载第三方类和获得类源码的实例由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.

使用反射动态加载第三方类 。

用反射加载第三方类用处在于: 使用XML或其他配文件配置要加载的类,从而和系统源代码分离。 对加载的类进行类检查,是加载的类符合自己定义的结构.

?
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
31
32
33
<?php
   abstract class Module {  #核心Module类库
     function baseFunc() {
       echo "I am baseFunc" ;
     }
    
     abstract function execute();
   }
  
   class ModuleRunner {
     private $configData = array(  #模拟xml配置,动态配置需要加载的Module
       "PersonModule" => array( "person" => "bob" ),
       "FtpModule" => array( "host" => "example.com" , "user" => "anon" )
     );
    
     private $modules = array();
    
     function init() {  #初始化ModuleRunner,加载配置中的Module
       $parent = new ReflectionClass( "Module" );
       foreach($ this ->configData as $moduleName => $params) {  #检查配置中的Module是否合法
         $moduleClass = new ReflectionClass($moduleName);
         if (! $moduleClass->isSubclassOf($parent)) {  #检查是否是Module的子类型
           throw new Exception( "unknown type : {$moduleName}" );
         }
         $module = $moduleClass->newInstance();
         foreach($moduleClass->getMethods() as $method) {  #检查配置中的函数的参数格式是否正确
           $ this ->handleMothod($module, $method, $params);
         }
         array_push($ this ->modules, $module);  #加载Module
       }
     }
    
     private function handleMothod(Module $module, ReflectionMethod $method, $params) {  #检查Module中的方法参数是

否和传入的$params名字相同,并且具有set方法    。

?
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
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
      $name = $method->getName();
       $args = $method->getParameters();
    
       if (count($args) != 1 || substr($name, 0 , 3 ) != "set" ) {  #如果没有配置中的类的方法的参数个数不为 1 ,或者方法名前 3 个字母不为set,返回 false
         return false ;
       }
      
       $property = strtolower(substr($name, 3 ));
       if (!isset($params[$property])) {  #如果方法名后三个字母与配置中的参数名不同,返回 false
         return false ;
       }
      
       $argClass = $args[ 0 ]->getClass();  #获取参数的类型
       if (empty($argClass)) {
         $method->invoke($module, $params[$property]);  #参数无类型限制则直接调用set方法
       } else {
         $method->invoke($module, $argClass->newInstance($params[$property]));  #有类型限制则新建一个实例并调用set方法
       }
     }
    
     public function getModules() {
       return $ this ->modules;
     }
   }
  
   class Person {  #第三方类
     public $name;
    
     function __construct($name) {
       $ this ->name = $name;
     }
   }
  
   class FtpModule extends Module {  #用户自定义第三方Module
     private $host = "default host" ;
     private $user = "default user" ;
    
     function setHost($host) {
       $ this ->host = $host;
     }
    
     function setUser($user) {
       $ this ->user = $user;
     }
    
     function execute() {
       echo "{$this->user} user {$this->host}" ;
     }
   }
  
   class PersonModule extends Module {  #用户自定义第三方Module
     private $person;
  
     function setPerson(Person $person) {
       $ this ->person = $person;
     }
    
     function execute() {
       if (isset($person)) {
         echo "I am {$this->person->name}" ;
       } else {
         echo "I am no user" ;
       }
     }
   }
  
   $modRunner = new ModuleRunner();
   $modRunner->init();
   var_dump($modRunner);
?>

输出 。

?
1
object(ModuleRunner)#1 (2) { ["configData":"ModuleRunner":private]=> array(2) { ["PersonModule"]=> array(1) { ["person"]=> string(3) "bob" } ["FtpModule"]=> array(2) { ["host"]=> string(11) "example.com" ["user"]=> string(4) "anon" } } ["modules":"ModuleRunner":private]=> array(2) { [0]=> object(PersonModule)#4 (1) { ["person":"PersonModule":private]=> object(Person)#10 (1) { ["name"]=> string(3) "bob" } } [1]=> object(FtpModule)#3 (2) { ["host":"FtpModule":private]=> string(11) "example.com" ["user":"FtpModule":private]=> string(4) "anon" } } }

通过反射获得类源码 。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
<?php
   function getSource(ReflectionClass $ref) {
     $path = $ref->getFileName();  #获取脚本文件文件名
     $file = file($path); #file()方法获取文件内容,并将内容保存在一个数组中,数组每个元素保存一行内容
     $start = $ref->getStartLine();  #获取类在脚本中的第一行行号
     $end = $ref->getEndLine();  #获取类在脚本中最后一行的行号
     $source = implode(array_slice($file, $start - 1 , $end - $start + 1 ));  #拼装类源码
    
     var_dump($source);
   }
 
   class Person {
     public $age;
     private $name;
    
     function say() {
       echo "yes" ;
     }
   }
  
   $ref = new ReflectionClass( "Person" );
   getSource($ref);
?>

最后此篇关于PHP通过反射动态加载第三方类和获得类源码的实例的文章就讲到这里了,如果你想了解更多关于PHP通过反射动态加载第三方类和获得类源码的实例的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com