PHP程序员小白到大牛集训(12期免息)

依赖容器解藕,外观模式facade统一调用

原创2018-12-02 15:12:30169
摘要:一、用依赖容器进行解藕<?php * 基本实现分三步: * 1.创建容器,将类与类的实例化过程绑定到容器中(不局限于类,也可是接口或其它)(用关联数组,类似操作数) * 2.服务注册,将可能用到的工具类全部绑定到容器中(初始化数组) * 3.容器依赖:或者叫依赖容器,调用工作类时直接传入容器对象即可,不用一个个调用对象,工具类的实例化由容器完成 //

一、用依赖容器进行解藕

<?php
* 基本实现分三步:
* 1.创建容器,将类与类的实例化过程绑定到容器中(不局限于类,也可是接口或其它)(用关联数组,类似操作数)
* 2.服务注册,将可能用到的工具类全部绑定到容器中(初始化数组)
* 3.容器依赖:或者叫依赖容器,调用工作类时直接传入容器对象即可,不用一个个调用对象,工具类的实例化由容器完成

//数据库操作类
class Db
{
   //数据库连接
   public function connect()
   {
      return '数据库连接成功<br>';
   }
}

//数据验证类
class Validate
{
   //数据验证
   public function check()
   {
      return '数据验证成功<br>';
   }
}

//视图图
class View
{
   //内容输出
   public function display()
   {
      return '用户登录成功';
   }
}

/******************************************************************************/

//一.创建容器类Container
class Container
{
   //创建属性,用空数组初始化,该属性用来保存类与类的实例化方法
    //创建空数组用来保存工具类以及实现方法
    //
   protected $instance = [];  //$instance['类名']='类的实例化过程(函数)';
    //public $instance = [];


    //初始化实例数组,将需要实例化的类,与实例化的方法进行绑定(bind绑定)
   public function bind($abstract, Closure $process) //(类的名称,类型/闭包,过程/函数)
   {
      //键名为类名,值为实例化的方法
      $this->instance[$abstract] = $process;

      /*执行类似这样
        $this->instance['db'] = function (){
            return new Db();
        };*/

   }

   //创建特定类的实例,
   public function make($abstract, $params=[])
   {
      return call_user_func_array($this->instance[$abstract],[]);
      //call_user_func_array执行函数,还可以执行特性的方法
   }

   //说明:
    /*执行类似这样
        $this->instance['db'] = function (){
            return new Db('127.0.0.1','root','123456');
        };
    ['db']等于$abstract;'127.0.0.1','root','123456'等于$params=[]
    */


}

/******************************************************************************/

//二、服务注册绑定: 调用容器中的bind()将类实例(对象)注册到容器中
$container = new Container();

//将Db类绑定到容器中
$container->bind('db', function(){
   return new Db();
});

//将Validate类实例绑定到容器中
$container->bind('validate', function(){
   return new Validate();
});

//将View类实例绑定到容器中
$container->bind('view', function(){
   return new View();
});

//测试:查看一下当前容器中的类实例
 //var_dump($container->instance); die;

 //public $instance = [];

 //array(3) { ["db"]=> object(Closure)#2 (0) { } ["validate"]=> object(Closure)#3 (0) { } ["view"]=> object(Closure)#4 (0) { } }

/******************************************************************************/

//三、容器依赖:将容器对象,以参数的方式注入到当前工作类中
//将所有用的的对象,以容器的方式,注入到当前工作类中

//用户类:工作类
class User
{
   //创建三个成员属性,用来保存本类所依赖的对象
   // protected $db = null;
   // protected $validate = null;
   // protected $view = '';
   //这三个与外部对象对应的三个属性可以全部删除了,因为它们都已经事先注册到了容器中

   //用户登录操作
   // public function login(Db $db, Validate $validate, View $view)
   //此时,只需从外部注入一个容器对象即可,Db,Validate和View实例方法全部封装到了容器中
   public function login(Container $container)
   {
      //实例化Db类并调用connect()连接数据库
      // $db = new Db();
      // echo $db->connect();
        //$container->make('db') 实例化Db类创建$db对象
      echo $container->make('db')->connect();

      //实例化Validate类并调用check()进行数据验证
      // $validate = new Validate();
      // echo $validate->check();
      echo $container->make('validate')->check();

      //实例化视图类并调用display()显示运行结果
      // $view = new View();
      echo $container->make('view')->display();
   }
}
//在客户端完成工具类的实例化(即工具类实例化前移)
// $db = new Db();
// $validate = new Validate();
// $view = new View();

//现在注入过程就非常简单了,只需要从外部注入一个容器对象即可

//创建User类
$user = new User();

//调用User对象的login方法进行登录操作
// echo $user->login();
// 将该类依赖的外部对象以参数方式注入到当前方法中,当然,推荐以构造器方式注入最方便
echo '<h3>用依赖容器进行解藕:</h3>';
// echo $user->login($db, $validate, $view);
//现在工作类中的login方法不需要再像对象依赖注入那样写三个对象了,只需要一个容器对象就可以了
echo $user->login($container);

用依赖容器进行解藕,调用者需要知道工作类中的对象是在哪个对应的类中;

用外观模式Facade可以解决统一接口的问题,调用者他不需要事先知道这工作类中的对象的在哪个具体的类中,他只需要调用Facade就可以执行工作类中的对象操作


二、外观模式/门面模式Facade

创建一个控制器container.php

<?php 

//数据库操作类
class Db
{
   //数据库连接
   public function connect()
   {
      return '数据库连接成功<br>';
   }
}

//数据验证类
class Validate
{
   //数据验证
   public function check()
   {
      return '数据验证成功<br>';
   }
}

//视图图
class View
{
   //内容输出
   public function display()
   {
      return '用户登录成功';
   }
}

/******************************************************************************/

//一.创建容器类
class Container
{
   //创建属性,用空数组初始化,该属性用来保存类与类的实例化方法
   public $instance = [];

   //初始化实例数组,将需要实例化的类,与实例化的方法进行绑定
   public function bind($abstract, Closure $process)
   {
      //键名为类名,值为实例化的方法
      $this->instance[$abstract] = $process;
   }

   //创建类实例
   public function make($abstract, $params=[])
   {
      return call_user_func_array($this->instance[$abstract],[]);
   }

}

/******************************************************************************/

//二、服务绑定: 将类实例注册到容器中
$container = new Container(); 

//将Db类绑定到容器中
$container->bind('db', function(){
   return new Db();
});

//将Validate类实例绑定到容器中
$container->bind('validate', function(){
   return new Validate();
});

//将View类实例绑定到容器中
$container->bind('view', function(){
   return new View();
});
<?php
/**
 * 外观模式:facade,也叫门面模式
 * 1.用一句来说:就是将操作进行封装,对外提供一个统一的接口
 * 2.因为操作可能分布在多个类中,而刚才学过的容器恰好可以将不同的类与实现封装起来
 * 3.所以外观模式facade与依赖容器是黄金搭档,经常会放在一起使用
 */

/**
 * 用户登录的操作涉及三个操作
 * 1.连接数据库
 * 2.用户数据验证
 * 3.输出提示信息
 * 对于调用者来说,他不需要事先知道这三个操作的在哪个具体的类中,他只需要调用Facade就可以执行以上三个操作
 */
 
 //可以在外观模型中使用初始化方法事先注入容器对象,来简化客户端调用
//在外观模式facade中,创建一个初始化方法事先注入容器,来简化代码
require 'container.php';

class Facade
{
   //创建成员属性保存容器对象
   protected static $container = null;

   //创建初始化方法为容器对象赋值
   public static function initialize(Container $container)
   {
      static::$container = $container;
   }

   /**
    * 因为已经在初始化方法中将容器对象导入到了当前类中,
    * 并且保存到了类的静态属性中,为所有类成员所共享,
    * 所以以下方法可直接调用不用再容器注入
    * 注意:这里全部采用后期静态延迟绑定方法来访问当前容器对象
    * 这主要是为了方便用户在静态继承的上下文环境中进行调用
    */
   //连接数据库
   public static function connect()
   {
      return static::$container->make('db')->connect();
   }

   //用户数据验证
   public static function check()
   {
      return static::$container->make('validate')->check();
   }

   //输出提示信息
   public static function display()
   {
      return static::$container->make('view')->display();
   }
}



//客户端调用

//初始化类门面类中的容器对象
Facade::initialize($container);

//静态统一调用内部的方法(无须重复注入依赖容器对象啦,实现了细节隐藏,通用性更强)
echo '<h3>简化后Facade外观模式的调用,不需要传入参数$container:</h3>';
echo Facade::connect();
echo Facade::check();
echo Facade::display();


批改老师:天蓬老师批改时间:2018-12-02 15:34:07
老师总结:门面技术是ThinkPHP5.1是容器的实现原理, 不过,真正的底层实现,还是要通过php的反射技术来实, 具体可以参考php.net中关于反射的知识点

发布手记

热门词条