zoukankan      html  css  js  c++  java
  • 控制反转( IoC)和依赖注入(DI)

    控制反转( IoC)和依赖注入(DI)

    tags: 容器 依赖注入 IOC DI 控制反转


    引言:如果你看过一些框架的源码或者手册,像是laravel或者tp5之类的,应该会提到容器,依赖注入,控制反转等词汇。或者是某些面试官会问到这类问题。希望这篇文章能让你有所收获。

    1.1、IoC(控制反转 Inversion of Control)

    简述:控制反转并不是一种技术,而是一种设计思想。通过控制反转容器(以后称容器),改变了原本某些对象运行时依赖其他对象资源时需要自己进行获取(比如通过new ClassName),所造成的对象之间的强耦合(耦合的概念如果不理解,可以先去了解一下。)。

      所谓IoC,对于程序来说,就是构造了一个容器,比如在tp5中,这个容器叫Container,此容器来负责控制对象和对象间的关系。在一个对象中,如果要使用另外的对象,就必须得到它(自己new一个),这样的话一个对象A和另一个对象B之间就有了很强的联系(也就是强耦合)。这种耦合体现在如果被依赖的资源类初始化的时候(也可能是其他时候)传入的参数变化了,你要修改两个类的代码,举例说明:
      
    你本来用的是php本身的session机制,现在想改成也支持redis,并在初始化的时候增加一个参数来控制。这个时候你不止需要去修改类Session类的代码,还要去修改类A的代码。部分代码如下

    //原代码
    class Session{
        function __construct(){
            //balabala
        }
    }
    
    class A{
        protected $driver;
        function __construct(){
           $this->driver = new Session();
        }
    }
    
    //需要增加一种redis方式后修改为
    class Session{
        protected $driver;
        function __construct(string $driver=''){
            if($driver === ''){
                //balabala
            }else if($driver === 'redis'){//增加了一种模式
                //balabala
            }else{
                //balabala
            }
            
        }
    }
    
    class A{
        protected $driver;
        function __construct(){
            $this->driver = new Session('redis');//我想使用redis了
        }
    }
    
    

    可以看到,不止我们修改了B的代码,还要去修改A的代码,这样如果依赖关系多起来的话,每次修改A依赖的对象,可能要处理A中很多条代码。会造成对象之间的强耦合。所以我们可以把A需要的东西,在A之前就创建出来,并通过构造函数参数传递给A。代码如下

     class A{
        protected $driver;
        function __construct(A $a){
            //我想使用redis了
        }
    }
    
    $driver = new Session('redis');//我想使用redis了
    $a = new A($driver)   
    

    这种方法下更改A依赖的所有对象都通过构造方法或者其他方法的形式给A,这些对象本身机制更改的时候就无需修改A的代码了。但是每次自己使用A的时候都去看看A需要些什么在前面都new一遍,感觉上很low啊,不如我们搞一个管家(容器),如果发现你需要什么,管家就给你什么。这样所依赖的类的创建都由容器来控制,也就是说控制对象创建的不再是引用它的对象,而是容器。对于某个具体的对象而言,以前是它控制其他对象,需要什么自己处理,现在是所有对象都被容器控制,所以控制反转是一种控制权的转移。

    1.2、DI(依赖注入 Dependency Injection Container)

      IoC的一个重点是在系统运行中,动态的向某个对象提供它所需要的其他对象。这一点是通过DI(Dependency Injection,依赖注入,或者叫依赖注入容器)来实现的。比如还是上面的例子,对象A需要操作Session,以前我们总是要在A中自己编写代码来获得一个Session对象,有了 容器这个管家我们就只需要告诉容器,A中需要一个Session对象,至于这个Session对象怎么构造,何时构造,A不需要知道。在系统运行时,容器会在适当的时候制造一个Session,通过构造方法注射到A当中,这样就完成了对各个对象之间关系的控制,而且这种关系是松耦合的。
      A需要依赖 Session才能正常运行,而这个Session是由容器注入到A中的,依赖注入的名字就这么来的。那么DI是如何实现的呢? php有一个高级特性是反射(reflection),原理这里大概说一下,

    反射可以在php运行中,提取出关于类、方法、属性、参数,注释等的详细信息,并可以动态的调用方法和类等。这种动态获取的信息以及动态调用对象的方法的功能称为反射API。

    反射允许程序在运行的时候动态的生成对象、执行对象的方法、改变对象的属性,容器就是通过反射来实现注入的。

    总结:
    控制反转是说创建对象的控制权进行转移,由原来的资源需求方,转移到了容器,依赖注入是说本来是资源需求方依赖资源,现在资源需求方依赖于容器对资源的注入,可以看出来依赖注入和控制反转说的其实是一个事情。

    顺便提一句,目前很多框架中都离不开反射功能。容器是一个典型的例子,容器在laravel和tp5中都是核心功能之一。

  • 相关阅读:
    docker学习之network:初识网络配置
    原来:HTTP可以复用TCP连接
    git tag的用法及意义
    Android,社招,面淘宝,指南【内部人员为你保驾护航】
    别了,拼多多!再也不想砍一刀了,哔哩哔哩 (゜-゜)つロ 干杯~
    【Android面试宝典】2021年,腾讯等大厂Android高级开发面试完全攻略!
    腾讯40岁老兵现身说法:35岁职业生涯分水岭,架构or管理,到底怎么选?
    【整理合集】Flutter 常用第三方库、插件、学习资料等
    [PAT]1011 World Cup Betting (20 分)Java
    [PAT] 1010 Radix (25 分)Java
  • 原文地址:https://www.cnblogs.com/vinter/p/8670032.html
Copyright © 2011-2022 走看看