全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

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

首先依赖注入和控制反转说的是同一个东西,是一种设计模式,这种设计模式用来减少程序间的耦合,鄙人学习了一下,看TP官网还没有相关的文章,就写下这篇拙作介绍一下这种设计模式,希望能为TP社区贡献一些力量。

首先先别追究这个设计模式的定义,否则你一定会被说的云里雾里,笔者就是深受其害,百度了N多文章,都是从理论角度来描述,充斥着大量的生涩词汇,要么就是java代码描述的,也生涩。

不管怎么样,总算弄清楚一些了,下面就以php的角度来描述一下依赖注入这个概念。

先假设我们这里有一个类,类里面需要用到数据库连接,按照最最原始的办法,我们可能是这样写这个类的:

class example {
  
  private $_db;
  function __construct(){
    include "./Lib/Db.php";
    $this->_db = new Db("localhost","root","123456","test");
  }
  function getList(){
    $this->_db->query("......");//这里具体sql语句就省略不写了
  }
 }

过程:

在构造函数里先将数据库类文件include进来;
然后又通过new Db并传入数据库连接信息实例化db类;
之后getList方法就可以通过$this->_db来调用数据库类,实现数据库操作。

看上去我们实现了想要的功能,但是这是一个噩梦的开始,以后example1,example2,example3....越来越多的类需要用到db组件,如果都这么写的话,万一有一天数据库密码改了或者db类发生变化了,岂不是要回头修改所有类文件?
ok,为了解决这个问题,工厂模式出现了,我们创建了一个Factory方法,并通过Factory::getDb()方法来获得db组件的实例:

class Factory {
  public static function getDb(){
    include "./Lib/Db.php";
    return new Db("localhost","root","123456","test");
  }
 }

sample类变成:

class example {
  
  private $_db;
  function __construct(){
    $this->_db = Factory::getDb();
  }
  function getList(){
    $this->_db->query("......");//这里具体sql语句就省略不写了
  }
 }

这样就完美了吗?再次想想一下以后example1,example2,example3....所有的类,你都需要在构造函数里通过Factory::getDb();获的一个Db实例,实际上由原来的直接与Db类的耦合变为了和Factory工厂类的耦合,工厂类只是帮你把数据库连接信息给包装起来了,虽然当数据库信息发生变化时只要修改Factory::getDb()方法就可以了,但是突然有一天工厂方法需要改名,或者getDb方法需要改名,你又怎么办?当然这种需求、、但有时候确实存在这种情况,一种解决方式是:

我们不从example类内部实例化Db组件,我们依靠从外部的注入,什么意思呢?看下面的例子:

class example {
  private $_db;
  function getList(){
    $this->_db->query("......");//这里具体sql语句就省略不写了
  }
  //从外部注入db连接
  function setDb($connection){
    $this->_db = $connection;
  }
 }
 //调用
$example = new example();
$example->setDb(Factory::getDb());//注入db连接
$example->getList();

这样一来,example类完全与外部类解除耦合了,你可以看到Db类里面已经没有工厂方法或Db类的身影了。我们通过从外部调用example类的setDb方法,将连接实例直接注入进去。这样example完全不用关心db连接怎么生成的了。
这就叫依赖注入,实现不是在代码内部创建依赖关系,而是让其作为一个参数传递,这使得我们的程序更容易维护,降低程序代码的耦合度,实现一种松耦合。

这还没完,我们再假设example类里面除了db还要用到其他外部类,我们通过:

$example->setDb(Factory::getDb());//注入db连接
$example->setFile(Factory::getFile());//注入文件处理类
$example->setImage(Factory::getImage());//注入Image处理类
 ...

我们没完没了的写这么多set?累不累?
ok,为了不用每次写这么多行代码,我们又去弄了一个工厂方法:

class Factory {
  public static function getExample(){
    $example = new example();
    $example->setDb(Factory::getDb());//注入db连接
    $example->setFile(Factory::getFile());//注入文件处理类
    $example->setImage(Factory::getImage());//注入Image处理类
    return $expample;
  }
 }

实例化example时变为:

$example=Factory::getExample();
$example->getList();

似乎完美了,但是怎么感觉又回到了上面第一次用工厂方法时的场景?这确实不是一个好的解决方案,所以又提出了一个概念:容器,又叫做IoC容器、DI容器。

我们本来是通过setXXX方法注入各种类,代码很长,方法很多,虽然可以通过一个工厂方法包装,但是还不是那么爽,好吧,我们不用setXXX方法了,这样也就不用工厂方法二次包装了,那么我们还怎么实现依赖注入呢?
这里我们引入一个约定:在example类的构造函数里传入一个名为Di $di的参数,如下:

class example {
  private $_di;
  function __construct(Di &$di){
    $this->_di = $di;
  }
  //通过di容器获取db实例
  function getList(){
    $this->_di->get('db')->query("......");//这里具体sql语句就省略不写了
  }
 }
$di = new Di();
$di->set("db",function(){
  return new Db("localhost","root","root","test"); 
 });
$example = new example($di);
$example->getList();

Di就是IoC容器,所谓容器就是存放我们可能会用到的各种类的实例,我们通过$di->set()设置一个名为db的实例,因为是通过回调函数的方式传入的,所以set的时候并不会立即实例化db类,而是当$di->get('db')的时候才会实例化,同样,在设计di类的时候还可以融入单例模式。

这样我们只要在全局范围内申明一个Di类,将所有需要注入的类放到容器里,然后将容器作为构造函数的参数传入到example,即可在example类里面从容器中获取实例。当然也不一定是构造函数,你也可以用一个 setDi(Di $di)的方法来传入Di容器,总之约定是你制定的,你自己清楚就行。

这样一来依赖注入以及关键的容器概念已经介绍完毕,剩下的就是在实际中使用并理解它吧!


# PHP  # 依赖注入  # DI  # 控制反转  # IoC  # 写了  # 这么多  # 方法来  # 就可以  # 的是  # 数据库类  # 是在  # 还没有  # 是一种  # 深受其害  # 还可以  # 你可以  # 也就  # 是这样  # 你也  # 才会  # 可以用  # 你自己  # 是从  # 提出了 


相关文章: 网站制作报价单模板图片,小松挖机官方网站报价?  如何用花生壳三步快速搭建专属网站?  如何处理“XML格式不正确”错误 常见XML well-formed问题解决方法  车管所网站制作流程,交警当场开简易程序处罚决定书,在交警网站查询不到怎么办?  大连企业网站制作公司,大连2025企业社保缴费网上缴费流程?  如何用PHP工具快速搭建高效网站?  教程网站设计制作软件,怎么创建自己的一个网站?  python的本地网站制作,如何创建本地站点?  建站之星如何助力网站排名飙升?揭秘高效技巧  购物网站制作公司有哪些,哪个购物网站比较好?  网站制作公司广州有几家,广州尚艺美发学校网站是多少?  义乌企业网站制作公司,请问义乌比较好的批发小商品的网站是什么?  建站之家VIP精选网站模板与SEO优化教程整合指南  如何在企业微信快速生成手机电脑官网?  装修招标网站设计制作流程,装修招标流程?  如何选择靠谱的建站公司加盟品牌?  如何选择高效便捷的WAP商城建站系统?  天河区网站制作公司,广州天河区如何办理身份证?需要什么资料有预约的网站吗?  建站之星微信建站一键生成小程序+多端营销系统  攀枝花网站建设,攀枝花营业执照网上怎么年审?  西安制作网站公司有哪些,西安货运司机用的最多的app或者网站是什么?  建站主机与服务器功能差异如何区分?  如何配置WinSCP新建站点的密钥验证步骤?  建站主机空间推荐 高性价比配置与快速部署方案解析  高端智能建站公司优选:品牌定制与SEO优化一站式服务  如何在IIS管理器中快速创建并配置网站?  小说建站VPS选用指南:性能对比、配置优化与建站方案解析  详解jQuery停止动画——stop()方法的使用  手机网站制作与建设方案,手机网站如何建设?  如何有效防御Web建站篡改攻击?  如何在Windows环境下新建FTP站点并设置权限?  如何快速生成高效建站系统源代码?  合肥制作网站的公司有哪些,合肥聚美网络科技有限公司介绍?  Python lxml的etree和ElementTree有什么区别  成都响应式网站开发,dw怎么把手机适应页面变成网页?  制作表格网站有哪些,线上表格怎么弄?  如何优化Golang Web性能_Golang HTTP服务器性能提升方法  微网站制作教程,不会写代码,不会编程,怎么样建自己的网站?  移动端手机网站制作软件,掌上时代,移动端网站的谷歌SEO该如何做?  如何在阿里云完成域名注册与建站?  建站之星如何快速生成多端适配网站?  定制建站方案优化指南:企业官网开发与建站费用解析  建站之星后台搭建步骤解析:模板选择与产品管理实操指南  公众号网站制作网页,微信公众号怎么制作?  建站与域名管理如何高效结合?  建站之星2.7模板快速切换与批量管理功能操作指南  盐城做公司网站,江苏电子版退休证办理流程?  高端建站如何打造兼具美学与转化的品牌官网?  制作网站建设的公司有哪些,网站建设比较好的公司都有哪些?  香港服务器租用每月最低只需15元? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。