热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

依赖注入解耦【PHP全栈开发经验分享】

1、什么是依赖注入依赖注入的本质就是在工作中的外部类实例化不在类中完成,而在类外完成

1、什么是依赖注入

依赖注入 的本质就是 在工作中的 外部类实例化不在类中完成,而在类外完成

/*** 用户登录* 1.涉及数据库操作,数据验证,模板输出* 2.分别对应Db类,Validate类,View类* 3.仅做演示,具体实例请同学们自行完成*///数据库操作类
class Db
{//数据库连接public function connect(){return '数据库连接成功
';}
}//数据验证类
class Validate
{//数据验证public function check(){return '数据验证成功
';}
}//视图图
class View
{//内容输出public function display(){return '用户登录成功';}
}//用户类
class User
{//用户登录操作public function login(){//实例化Db类并调用connect()连接数据库$db = new Db();echo $db->connect();//实例化Validate类并调用check()进行数据验证$validate = new Validate();echo $validate->check();//实例化视图类并调用display()显示运行结果$view = new View();echo $view->display();}
}/*** 知识:什么是客户端* 客户端:只要能发起请求,都可以看作客户端,浏览器,一段代码都可以* 以下的代码,在实例化User类,并调用其内部的loign方法进行工作* 所以,$user = new User();就是客户端代码* 或者,也可以这样理解,凡写在类或函数等代码之外的,都可看作客户端*/
//创建User类
$user = new User();//调用User对象的login方法进行登录操作
echo $user->login();/*** 存在的问题:* 以上的四个类,只有User是实际工作类,其它三个都是工具类(Db,Validate,View)* 1.工作类中调用的工具类一旦发生变化,必须修改对这些工具类的所有引用代码,例如Db参数变化* 2.工作类的调用者必须对要用到的所有工具类,非常熟悉,对参数与返回值必须了解* 3.工作类对以上三个工具类,形成了严重的依赖,也叫类之间严重耦合** 下面我们通过最常用的依赖注入(DI)来解藕*/

 

class Db
{//数据库连接public function connect(){return '数据库连接成功
';}
}//数据验证类
class Validate
{//数据验证public function check(){return '数据验证成功
';}
}//视图图
class View
{//内容输出public function display(){return '用户登录成功';}
}//用户类
class User
{//用户登录操作public function login(Db $db, Validate $validate, View $view) //对象变量传入进来,Db,Validate,View 为类{//实例化Db类并调用connect()连接数据库// $db = new Db();echo $db->connect();//实例化Validate类并调用check()进行数据验证// $validate = new Validate();echo $validate->check();//实例化视图类并调用display()显示运行结果// $view = new View();echo $view->display();}
}
//在客户端完成工具类的实例化(即工具类实例化前移)
$db = new Db();
$validate = new Validate();
$view = new View();//创建User类
$user = new User();//调用User对象的login方法进行登录操作
// echo $user->login();
// 将该类依赖的外部对象以参数方式注入到当前方法中,当然,推荐以构造器方式注入最方便
echo '

用依赖注入进行解藕:

';
echo $user->login($db, $validate, $view);/*** 虽然将依赖类的实例化前移到客户端,但解决了类之间的依赖问题* 但是仍存在以下几个问题:* 1.为了使工作类User正常工具,必须事先在外部将所需要的类全部事先实例化;* 2.只要涉及实例化,就要求客户端(调用者)必须对这些依赖类的细节非常了解,例如参数与返回值** 那么能不能让用户把实例化依赖类的步骤都省略掉呢?这样岂不是更好,更简单* 我们调用外部依赖类,只要给一个类名,以及一个创建该类实例的方法(构造器)就可以了呢?* 即: 我们只给出: 类名, 创建类实例的方法,其它一概不管* 下面我们通过的"容器技术"来这现这种傻瓜式的的解藕过程*/// 构造器中实现注入以简化代码
//用户类
class User
{protected $db = null;protected $validate = null;protected $view = null;public function __construct(Db $db, Validate $validate, View $view){$this->db = $db;$this->validate = $validate;$this->view = $view;}//用户登录操作public function login(){//实例化Db类并调用connect()连接数据库echo $this->db->connect();//实例化Validate类并调用check()进行数据验证echo $this->validate->check();//实例化视图类并调用display()显示运行结果echo $this->view->display();}
}

构造器注入

class Db
{//数据库连接public function connect(){return '数据库连接成功
';}
}//数据验证类
class Validate
{//数据验证public function check(){return '数据验证成功
';}
}//视图图
class View
{//内容输出public function display(){return '用户登录成功';}
}//用户类
class User
{//用户登录操作public function login(Db $db, Validate $validate, View $view) //对象变量传入进来,Db,Validate,View 为类{//实例化Db类并调用connect()连接数据库// $db = new Db();echo $db->connect();//实例化Validate类并调用check()进行数据验证// $validate = new Validate();echo $validate->check();//实例化视图类并调用display()显示运行结果// $view = new View();echo $view->display();}
}
//在客户端完成工具类的实例化(即工具类实例化前移)
$db = new Db();
$validate = new Validate();
$view = new View();//创建User类
$user = new User();//调用User对象的login方法进行登录操作
// echo $user->login();
// 将该类依赖的外部对象以参数方式注入到当前方法中,当然,推荐以构造器方式注入最方便
echo '

用依赖注入进行解藕:

';
echo $user->login($db, $validate, $view);/*** 虽然将依赖类的实例化前移到客户端,但解决了类之间的依赖问题* 但是仍存在以下几个问题:* 1.为了使工作类User正常工具,必须事先在外部将所需要的类全部事先实例化;* 2.只要涉及实例化,就要求客户端(调用者)必须对这些依赖类的细节非常了解,例如参数与返回值** 那么能不能让用户把实例化依赖类的步骤都省略掉呢?这样岂不是更好,更简单* 我们调用外部依赖类,只要给一个类名,以及一个创建该类实例的方法(构造器)就可以了呢?* 即: 我们只给出: 类名, 创建类实例的方法,其它一概不管* 下面我们通过的"容器技术"来这现这种傻瓜式的的解藕过程*/// 构造器中实现注入以简化代码
//用户类
class User
{protected $db = null;protected $validate = null;protected $view = null;public function __construct(Db $db, Validate $validate, View $view){$this->db = $db;$this->validate = $validate;$this->view = $view;}//用户登录操作public function login(){//实例化Db类并调用connect()连接数据库echo $this->db->connect();//实例化Validate类并调用check()进行数据验证echo $this->validate->check();//实例化视图类并调用display()显示运行结果echo $this->view->display();}
}

class Db
{//数据库连接public function connect(){return '数据库连接成功
';}
}//数据验证类
class Validate
{//数据验证public function check(){return '数据验证成功
';}
}//视图图
class View
{//内容输出public function display(){return '用户登录成功';}
}/******************************************************************************///一.创建容器类
class Container
{//创建属性,用空数组初始化,该属性用来保存类与类的实例化方法public $instance = [];//初始化实例数组,将需要实例化的类,与实例化的方法进行绑定Closure $process闭包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类绑定到容器中 function(){return new Db();}实际是类实例化后的方法
$container->bind('db', function(){return new Db();
});//将Validate类实例绑定到容器中
$container->bind('validate', function(){return new Validate();
});//将View类实例绑定到容器中
$container->bind('view', function(){return new View();
});

/*** 外观模式:facade,也叫门面模式* 1.用一句来说:就是将操作进行封装,对外提供一个统一的接口* 2.因为操作可能分布在多个类中,而刚才学过的容器恰好可以将不同的类与实现封装起来* 3.所以外观模式与依赖容器是黄金搭档,经常会放在一起使用*//*** 用户登录的操作涉及三个操作* 1.连接数据库* 2.用户数据验证* 3.输出提示信息*/require 'container.php';//创建Facade类,实现以上三个功能
class Facade
{//连接数据库public static function connect(Container $container){return $container->make('db')->connect();}//用户数据验证public static function check(Container $container){return $container->make('validate')->check();}//输出提示信息public static function display(Container $container){return $container->make('view')->display();}
}//客户端调用
echo Facade::connect($container);
echo Facade::check($container);
echo Facade::display($container);//可以在外观模型中使用初始化方法事先注入容器对象,来简化客户端调用

 


推荐阅读
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 阿,里,云,物,联网,net,core,客户端,czgl,aliiotclient, ... [详细]
  • 本文探讨了C语言中指针的应用与价值,指针在C语言中具有灵活性和可变性,通过指针可以操作系统内存和控制外部I/O端口。文章介绍了指针变量和指针的指向变量的含义和用法,以及判断变量数据类型和指向变量或成员变量的类型的方法。还讨论了指针访问数组元素和下标法数组元素的等价关系,以及指针作为函数参数可以改变主调函数变量的值的特点。此外,文章还提到了指针在动态存储分配、链表创建和相关操作中的应用,以及类成员指针与外部变量的区分方法。通过本文的阐述,读者可以更好地理解和应用C语言中的指针。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • 猜字母游戏
    猜字母游戏猜字母游戏——设计数据结构猜字母游戏——设计程序结构猜字母游戏——实现字母生成方法猜字母游戏——实现字母检测方法猜字母游戏——实现主方法1猜字母游戏——设计数据结构1.1 ... [详细]
  • 集合的遍历方式及其局限性
    本文介绍了Java中集合的遍历方式,重点介绍了for-each语句的用法和优势。同时指出了for-each语句无法引用数组或集合的索引的局限性。通过示例代码展示了for-each语句的使用方法,并提供了改写为for语句版本的方法。 ... [详细]
  • Spring常用注解(绝对经典),全靠这份Java知识点PDF大全
    本文介绍了Spring常用注解和注入bean的注解,包括@Bean、@Autowired、@Inject等,同时提供了一个Java知识点PDF大全的资源链接。其中详细介绍了ColorFactoryBean的使用,以及@Autowired和@Inject的区别和用法。此外,还提到了@Required属性的配置和使用。 ... [详细]
  • Week04面向对象设计与继承学习总结及作业要求
    本文总结了Week04面向对象设计与继承的重要知识点,包括对象、类、封装性、静态属性、静态方法、重载、继承和多态等。同时,还介绍了私有构造函数在类外部无法被调用、static不能访问非静态属性以及该类实例可以共享类里的static属性等内容。此外,还提到了作业要求,包括讲述一个在网上商城购物或在班级博客进行学习的故事,并使用Markdown的加粗标记和语句块标记标注关键名词和动词。最后,还提到了参考资料中关于UML类图如何绘制的范例。 ... [详细]
  • 本文介绍了在MFC下利用C++和MFC的特性动态创建窗口的方法,包括继承现有的MFC类并加以改造、插入工具栏和状态栏对象的声明等。同时还提到了窗口销毁的处理方法。本文详细介绍了实现方法并给出了相关注意事项。 ... [详细]
  • 本文介绍了H5游戏性能优化和调试技巧,包括从问题表象出发进行优化、排除外部问题导致的卡顿、帧率设定、减少drawcall的方法、UI优化和图集渲染等八个理念。对于游戏程序员来说,解决游戏性能问题是一个关键的任务,本文提供了一些有用的参考价值。摘要长度为183字。 ... [详细]
  • STL迭代器的种类及其功能介绍
    本文介绍了标准模板库(STL)定义的五种迭代器的种类和功能。通过图表展示了这几种迭代器之间的关系,并详细描述了各个迭代器的功能和使用方法。其中,输入迭代器用于从容器中读取元素,输出迭代器用于向容器中写入元素,正向迭代器是输入迭代器和输出迭代器的组合。本文的目的是帮助读者更好地理解STL迭代器的使用方法和特点。 ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • PHP中的单例模式与静态变量的区别及使用方法
    本文介绍了PHP中的单例模式与静态变量的区别及使用方法。在PHP中,静态变量的存活周期仅仅是每次PHP的会话周期,与Java、C++不同。静态变量在PHP中的作用域仅限于当前文件内,在函数或类中可以传递变量。本文还通过示例代码解释了静态变量在函数和类中的使用方法,并说明了静态变量的生命周期与结构体的生命周期相关联。同时,本文还介绍了静态变量在类中的使用方法,并通过示例代码展示了如何在类中使用静态变量。 ... [详细]
  • 本文讨论了微软的STL容器类是否线程安全。根据MSDN的回答,STL容器类包括vector、deque、list、queue、stack、priority_queue、valarray、map、hash_map、multimap、hash_multimap、set、hash_set、multiset、hash_multiset、basic_string和bitset。对于单个对象来说,多个线程同时读取是安全的。但如果一个线程正在写入一个对象,那么所有的读写操作都需要进行同步。 ... [详细]
  • C++语言入门:数组的基本知识和应用领域
    本文介绍了C++语言的基本知识和应用领域,包括C++语言与Python语言的区别、C++语言的结构化特点、关键字和控制语句的使用、运算符的种类和表达式的灵活性、各种数据类型的运算以及指针概念的引入。同时,还探讨了C++语言在代码效率方面的优势和与汇编语言的比较。对于想要学习C++语言的初学者来说,本文提供了一个简洁而全面的入门指南。 ... [详细]
author-avatar
无限的天空-空间
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有