依赖注入又叫控制反转,使用过框架的人应该都不陌生。很多人一看名字就觉得是非常高大上的东西,就对它望而却步,今天抽空研究了下,解开他它的神秘面纱。废话不多说,直接上代码;

  

  /*  *
    *
    * 工具类,使用该类来实现自动依赖注入。
    *
  */
  class Ioc {

    // 获得类的对象实例
    public static function getInstance($className) {

    $paramArr = self::getMethodParams($className);

    return (new ReflectionClass($className))->newInstanceArgs($paramArr);
  }

  /**
    * 执行类的方法
    * @param [type] $className [类名]
    * @param [type] $methodName [方法名称]
    * @param [type] $params [额外的参数]
    * @return [type] [description]
  */
  public static function make($className, $methodName, $params = []) {

    // 获取类的实例
    $instance = self::getInstance($className);

    // 获取该方法所需要依赖注入的参数
    $paramArr = self::getMethodParams($className, $methodName);

    return $instance->{$methodName}(...array_merge($paramArr, $params));
  }

  /**
    * 获得类的方法参数,只获得有类型的参数
    * @param [type] $className [description]
    * @param [type] $methodsName [description]
    * @return [type] [description]
  */
  protected static function getMethodParams($className, $methodsName = '__construct') {

    // 通过反射获得该类
    $class = new ReflectionClass($className);
    $paramArr = []; // 记录参数,和参数类型

    // 判断该类是否有构造函数
    if ($class->hasMethod($methodsName)) {
      // 获得构造函数
      $construct = $class->getMethod($methodsName);

      // 判断构造函数是否有参数
      $params = $construct->getParameters();

      if (count($params) > 0) {

      // 判断参数类型
      foreach ($params as $key => $param) {

        if ($paramClass = $param->getClass()) {

          // 获得参数类型名称
          $paramClassName = $paramClass->getName();

          // 获得参数类型
          $args = self::getMethodParams($paramClassName);
          $paramArr[] = (new ReflectionClass($paramClass->getName()))->newInstanceArgs($args);
        }
      }
    }
  }

  return $paramArr;
}
}

上面的代码使用php的反射函数,创建了一个容器类,使用该类来实现其他类的依赖注入功能。上面的依赖注入分为两种,一种是构造函数的依赖注入,一种是方法的依赖注入。 我们使用下面三个类来做下测试。
  class A {

    protected $cObj;

    /**
      * 用于测试多级依赖注入 B依赖A,A依赖C
      * @param C $c [description]
    */
    public function __construct(C $c) {

      $this->cObj = $c;
    }

    public function aa() {

      echo 'this is A->test';
    }

    public function aac() {

      $this->cObj->cc();
    }
  }

  class B {

    protected $aObj;

  /**
    * 测试构造函数依赖注入
    * @param A $a [使用引来注入A]
  */
  public function __construct(A $a) {

    $this->aObj = $a;
  }

  /**
    * [测试方法调用依赖注入]
    * @param C $c [依赖注入C]
    * @param string $b [这个是自己手动填写的参数]
    * @return [type] [description]
  */
  public function bb(C $c, $b) {

    $c->cc();
    echo "\r\n";

    echo 'params:' . $b;
  }

  /**
    * 验证依赖注入是否成功
    * @return [type] [description]
  */
  public function bbb() {

    $this->aObj->aac();
  }
}

  class C {

    public function cc() {

    echo 'this is C->cc';
  }
}

测试构造函数的依赖注入
// 使用Ioc来创建B类的实例,B的构造函数依赖A类,A的构造函数依赖C类。
$bObj = Ioc::getInstance('B');
$bObj->bbb(); // 输出:this is C->cc , 说明依赖注入成功。

// 打印$bObj
var_dump($bObj);

// 打印结果,可以看出B中有A实例,A中有C实例,说明依赖注入成功。
object(B)#3 (1) {
  ["aObj":protected]=>
  object(A)#7 (1) {
    ["cObj":protected]=>
    object(C)#10 (0) {
    }
  }
}

测试方法依赖注入

Ioc::make('B', 'bb', ['this is param b']);

// 输出结果,可以看出依赖注入成功。
this is C->cc
params:this is param b

从上面两个例子可以看出我们创建对象或者调用方法时,根本就不用知道该类或该方法依赖了那个类。使用反射机制可以轻松的为我们自动注入所需要的类。
总结

好了,看到上面的代码是不是觉得很简单,其实只要熟悉php的反射机制,依赖注入并不难实现,上面的代码为了方便理解,所以写的简单除暴,在实际的项目中肯定不会这么简单,比如:会对注入的类和参数进行配置,比如会缓存实例化过的类,下次需要该类的实例时,可以直接使用,而不用在重新初始化,等等。不过相信原理了解了,其他的可以随着项目的需求自己去完善。

  

PHP反射机制实现自动依赖注入的更多相关文章

  1. 新项目升级到JFinal3.5之后的改变-着重体验自动依赖注入

    最近,JFinal3.5发布,喜大普奔,我也应JBolt用户的需求,将JBolt进行了升级,实现可配置自动注入开启,支持JFinal3.5的项目生成.具体可以看:JBolt升级日志 这等工作做完后,我 ...

  2. 几十行代码实现ASP.NET Core自动依赖注入

    在开发.NET Core web服务的时候,我们习惯使用自带的依赖注入容器来进行注入. 于是就会经常进行一个很频繁的的重复动作:定义一个接口->写实现类->注入 有时候会忘了写Add这一步 ...

  3. 前端axios发送的数据后端接收不到(没有自动依赖注入)可能的原因

    前端请求头content-type没有进行正确设置,后端无法解析该类型数据,比如说: 后端若想接收json类型的数据,则需要配置相应的转换器,(spring中可使用@RequestBody注解),若没 ...

  4. JavaBean 反射机制实现自动配置数据

    声明:该版本是没完成的.该博文只做记录代码用 String memberType(String name) throws Exception { return getType(getClass().g ...

  5. 学习Spring——依赖注入

    前言: 又开始动笔开了“学习Spring”系列的头…… 其实一开始写“学习SpringMVC”的几篇文章是出于想系统的了解下Spring以及SpringMVC,因为平时在公司中虽然每天都在使用Spri ...

  6. 转载--浅谈spring4泛型依赖注入

    转载自某SDN-4O4NotFound Spring 4.0版本中更新了很多新功能,其中比较重要的一个就是对带泛型的Bean进行依赖注入的支持.Spring4的这个改动使得代码可以利用泛型进行进一步的 ...

  7. 理解依赖注入(IOC)和学习Unity

    资料1: IOC:英文全称:Inversion of Control,中文名称:控制反转,它还有个名字叫依赖注入(Dependency Injection). 作用:将各层的对象以松耦合的方式组织在一 ...

  8. spring学习总结一----控制反转与依赖注入

    spring作为java EE中使用最为广泛的框架,它的设计体现了很多设计模式中经典的原则和思想,所以,该框架的各种实现方法非常值得我们去研究,下面先对spring中最为重要的思想之一----控制反转 ...

  9. php+laravel依赖注入浅析

    laravel容器包含控制反转和依赖注入,使用起来就是,先把对象bind好,需要时可以直接使用make来取就好. 通常我们的调用如下. $config = $container->make('c ...

随机推荐

  1. windows2008系统缓存修改

    Windwos Server 2008 中经常有物理内存占用较高,但任务管理器中各进程的内存看起来占用并不多情况. 这可能是由于OS为保证服务器性能不限制系统缓存,服务器系统长时间执行I/O,系统文件 ...

  2. pandoc 基本使用

    pandoc –s 输入文件.后缀 –o 输出文件.后缀

  3. SQL优化系列——查询优化器

    大多数查询优化器将查询计划用“计划节点”树表示.计划节点封装执行查询所需的单个操作.节点被布置为树,中间结果从树的底部流向顶部.每个节点具有零个或多个子节点 - 这些子节点是输出作为父节点输入的节点. ...

  4. 安卓权威编程指南 - 第五章学习笔记(两个Activity)

    学习安卓编程权威指南第五章的时候自己写了个简单的Demo来加深理解两个Activity互相传递数据的问题,然后将自己的学习笔记贴上来,如有错误还请指正. IntentActivityDemo学习笔记 ...

  5. Zabbix 告警内容配置

    #配置媒介告警类型 #----------------------------------------------------------------------------------------- ...

  6. BloomFilter理解

    知道BloomFilter是因为RocksDB数据库中用到了这个技术,用于判断1个数据是否存在于1个SST文件中. BloomFilter可能存在误判,就是判断数据是存在集合中,而实际上可能不存在,概 ...

  7. HDOJ 2020 绝对值排序

    #include<iostream> #include<cmath> #include<algorithm> #include<vector> usin ...

  8. C#一年中有多少周方法和js一年中第几周

    最近在做一个时间插件,用的是jquery-daterangepicker ,现在分享一下查询时间是一年中的第几周的js方法  和 一年中有多少周的C#后台方法,默认是按照周一为一周的开始,如果一年的第 ...

  9. java加减天数

    指定日期的加减天数 public static String setDay(int num,String newDate) throws ParseException{ SimpleDateForma ...

  10. servlet的执行过程简介(从tomcat服务器和web应用的角度)

    该链接详解htttp请求和响应 http://www.cnblogs.com/goxcheer/p/8424175.html 1.web应用工程发布到tomcat服务器 2.客户端访问某个web资源, ...