依赖注入 DI 控制反转 IOC MD
| Markdown版本笔记 | 我的GitHub首页 | 我的博客 | 我的微信 | 我的邮箱 |
|---|---|---|---|---|
| MyAndroidBlogs | baiqiantao | baiqiantao | bqt20094 | baiqiantao@sina.com |
依赖注入 DI 控制反转 IOC
目录
依赖注入 Dependency injection
这里通过一个简单的案例来说明。
在公司里有一个常见的案例:"把任务指派个程序员完成"。
把这个案例用面向对象(OO)的方式来设计,我们创建两个类:Task 和 Phper (php 程序员)
Step1 设计
public class Phper {
private String name;
public Phper(String name){
this.name=name;
}
public void writeCode(){
System.out.println(this.name + " is writing php code");
}
}
public class Task {
private String name;
private Phper owner;
public Task(String name){
this.name =name;
this.owner = new Phper("zhang3");//关键看这里
}
public void start(){
System.out.println(this.name+ " started");
this.owner.writeCode();
}
}
测试:
public class MyFramework {
public static void main(String[] args) {
Task t = new Task("Task #1");
t.start();
}
}
运行结果:
Task #1 started
zhang3is writing php code
我们看一看这个设计有什么问题。
如果同事仰慕你的设计,要重用你的代码,你把程序打成一个类库(jar包)发给同事。
现在问题来了:同事发现这个Task 类 和 程序员 zhang3 绑定在一起,他所有创建的Task,都是程序员zhang3负责,他要把一些任务指派给Lee4, 就需要修改Task的源程序, 如果没有Task的源程序,就无法把任务指派给他人。
而类库(jar包)的使用者通常不需要也不应该来修改类库的源码,我们很自然的想到,应该让用户来指派任务负责人,于是有了新的设计。
Step2 设计
Phper不变。
public class Task {
private String name;
private Phper owner;
public Task(String name){
this.name =name;
}
public void setOwner(Phper owner){//关键看这里。用户可在使用时按需指派特定的PHP程序员
this.owner = owner;
}
public void start(){
System.out.println(this.name+ " started");
this.owner.writeCode();
}
}
测试:
public class MyFramework {
public static void main(String[] args) {
Task t = new Task("Task #1");
Phper owner = new Phper("lee4");
t.setOwner(owner);//用户在使用时按需指派特定的PHP程序员
t.start();
}
}
这样用户就可在使用时指派特定的PHP程序员。
我们知道,Task类依赖Phper类,之前,Task类绑定特定的实例,现在这种依赖可以在使用时按需绑定,这就是依赖注入(DI)。
这个例子,我们通过方法setOwner注入依赖对象,另外一个常见的注入办法是在Task的构造函数注入:
public Task(String name,Phper owner){
this.name = name;
this.owner = owner;
}
在Java开发中,把一个对象实例传给一个新建对象的情况十分普遍,通常这就是注入依赖,Step2 的设计实现了依赖注入。
我们来看看Step2 的设计有什么问题。
如果公司是一个单纯使用PHP的公司,所有开发任务都有Phper 来完成,这样这个设就已经很好了,不用优化。但是随着公司的发展,有些任务需要JAVA来完成,公司招了写Javaer(java程序员),现在问题来了,这个Task类库的的使用者发现,任务只能指派给Phper,一个很自然的需求就是,Task应该即可指派给Phper也可指派给Javaer。
Step3 设计
我们发现不管Phper 还是 Javaer 都是Coder(程序员), 把Task类对Phper类的依赖改为对 Coder 的依赖即可。
新增Coder接口
public interface Coder {
void writeCode();
}
修改Phper类实现Coder接口
public class Phper implements Coder {
private String name;
public Phper(String name){
this.name=name;
}
@Override
public void writeCode(){
System.out.println(this.name + " is writing php code");
}
}
新类Javaer实现Coder接口
public class Javaer implements Coder {
private String name;
public Javaer(String name){
this.name=name;
}
@Override
public void writeCode(){
System.out.println(this.name + " is writing Java code");
}
}
修改Task由对Phper类的依赖改为对Coder的依赖
public class Task {
private String name;
private Coder owner;
public Task(String name) {
this.name = name;
}
public void setOwner(Coder owner) {
this.owner = owner;
}
public void start() {
System.out.println(this.name + " started");
this.owner.writeCode();
}
}
测试
public class MyFramework {
public static void main(String[] args) {
Task t = new Task("Task #1");
Coder owner = new Phper("lee4");
//Coder owner = new Javaer("Wang5");
t.setOwner(owner);
t.start();
}
}
现在用户可以和方便的把任务指派给Javaer 了,如果有新的Pythoner加入,没问题,类库的使用者只需让Pythoner实现Coder接口,就可把任务指派给Pythoner, 无需修改Task 源码, 提高了类库的可扩展性。
总结
回顾一下,我们开发的Task类:
- 在Step1 中,Task与特定实例绑定(zhang3 Phper)
- 在Step2 中,Task与特定类型绑定(Phper)
- 在Step3 中,Task与特定接口绑定(Coder)
虽然都是绑定, 从Step1,Step2 到 Step3 灵活性、可扩展性是依次提高的。
Step1 作为反面教材不可取, 至于是否需要从Step2 提升为Step3, 要看具体情况。
依赖注入(DI)实现了控制反转(IoC)的思想,看看怎么反转的:
- Step1 设计中 任务Task依赖负责人owner, 所以就主动新建一个 Phper 赋值给owner,这里可能是新建,也可能是在容器中获取一个现成的Phper,是新建还是获取无关紧要,关键是主动赋值。
- 在Step2 和 Step3中, Task 的 owner 是被动赋值的,谁来赋值,Task自己不关心,可能是类库的用户,也可能是框架或容器,Task交出赋值权,从主动赋值到被动赋值, 这就是控制反转。
控制反转 Inversion Of Control
什么是控制反转 ?
简单的说,从主动变被动就是控制反转。
上文以依赖注入的例子,对控制反转做了个简单的解释。
控制反转是一个很广泛的概念, 依赖注入是控制反转的一个例子,但控制反转的例子还很多,甚至与软件开发无关。
传统的程序开发,人们总是从main 函数开始,调用各种各样的库来完成一个程序。这样的开发,开发者控制着整个运行过程。
而现在人们使用框架(Framework)开发,使用框架时,框架控制着整个运行过程。
对比以下的两个简单程序。
1、简单java程序
public class Activity {
public Activity() {
this.onCreate();//开发者主动调用onCreate()方法
}
public void onCreate() {
System.out.println("onCreate called");
}
public void sayHi() {
System.out.println("Hello world!");
}
public static void main(String[] args) {
Activity a = new Activity();
a.sayHi();
}
}
2、简单Android程序
public class MainActivity extends Activity {
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
System.out.println("onCreate called");
}
public void sayHi(View view) {//某个View的点击事件
System.out.println("Hello world!");
}
}
这两个程序最大的区别就是,前者程序的运行完全由开发控制,后者程序的运行由Android框架控制。
虽然两个程序都有个onCreate方法,但在前者程序中,如果开发者觉得onCreate名称不合适,想改为Init,没问题,直接就可以改; 相比下,后者的onCreate名称就不能修改,因为后者使用了框架,享受框架带来福利的同时,就要遵循框架的规则。
这就是控制反转。
可以说, 控制反转是所有框架最基本的特征,也是框架和普通类库最大的不同点。
通过框架可以把许多共用的逻辑放到框架里,让用户专注自己程序的逻辑,这也是为什么现在,无论手机开发,网页开发,还是桌面程序, 也不管是Java、PHP还是Python,框架无处不在。
控制反转还有一个漂亮的比喻,好莱坞原则(Hollywood principle):
"don't call us, we'll call you。"
"不要打电话给我们,我们会打给你(如果合适)"
这是好莱坞电影公司对面试者常见的答复。
事实上,不只电影行业,基本上所有公司人力资源部对面试者都会说类似的话,让面试者从主动联系转换为被动等待。
JAVA中依赖注入的几种方式
通过set方法注入
public class ClassA {
ClassB classB;
public void setClassB(ClassB b) {
classB = b;
}
}
通过构造器注入
public class ClassA {
ClassB classB;
public void ClassA(ClassB b) {
classB = b;
}
}
通过注解注入
public class ClassA {
@inject ClassB classB;//此时并不会完成注入,还需要依赖注入框架的支持,如RoboGuice,Dagger2
...
public ClassA() {}
}
通过接口注入,形式一:
interface InterfaceB {
void doIt();
}
public class ClassA {
InterfaceB clzB;
public void doSomething() {
clzB = (InterfaceB) Class.forName("...").newInstance();//根据预先在配置文件中设定的实现类的类名动态加载实现类
clzB.doIt();
}
}
此种接口注入方式因为具备侵入性,它要求组件必须与特定的接口相关联,因此实际使用有限。
通过接口注入,形式二:
public interface InjectB {
void injectB(ClassB b);
}
class A implements InjectB {
ClassB classB;
@Override
public void injectB(ClassB b) {
classB = b;
}
}
2017-9-14
依赖注入 DI 控制反转 IOC MD的更多相关文章
- ADO.NET .net core2.0添加json文件并转化成类注入控制器使用 简单了解 iTextSharp实现HTML to PDF ASP.NET MVC 中 Autofac依赖注入DI 控制反转IOC 了解一下 C# AutoMapper 了解一下
ADO.NET 一.ADO.NET概要 ADO.NET是.NET框架中的重要组件,主要用于完成C#应用程序访问数据库 二.ADO.NET的组成 ①System.Data → DataTable, ...
- 依赖注入 DI 控制反转 IOC 概念 案例 MD
Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱 MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina ...
- 浅析“依赖注入(DI)/控制反转(IOC)”的实现思路
开始学习Spring的时候,对依赖注入(DI)——也叫控制反转(IOC)—— 的理解不是很深刻.随着学习的深入,也逐渐有了自己的认识,在此记录,也希望能帮助其他入门同学更深入地理解Spring.本文不 ...
- ASP.NET MVC 中 Autofac依赖注入DI 控制反转IOC 了解一下
先简单了解一这个几个 名词的意思. 控制反转(IOC) 依赖注入(DI) 并不是某种技术. 而是一种思想.一种面向对象编程法则 什么是控制反转(IOC)? 什么是依赖注入(DI) 可以点击下面链接 ...
- ASP.NET中IOC容器Autofac(依赖注入DI 控制反转IOC)
IOC的一个重点是在程序运行中,动态的向某个对象提供它所需要的其他对象.这一点是通过DI来实现的.Autofac则是比较流行的一款IOC容器. IoC和DI有什么关系呢?其实它们是同一个概念的不同角度 ...
- 轻松学,浅析依赖倒置(DIP)、控制反转(IOC)和依赖注入(DI) 依赖注入和控制反转的理解,写的太好了。
轻松学,浅析依赖倒置(DIP).控制反转(IOC)和依赖注入(DI) 2017年07月13日 22:04:39 frank909 阅读数:14269更多 所属专栏: Java 反射基础知识与实战 ...
- 浅谈(IOC)依赖注入与控制反转(DI)
前言:参考了百度文献和https://www.cnblogs.com/liuqifeng/p/11077592.html以及http://www.cnblogs.com/leoo2sk/archive ...
- Spring Framework------>version4.3.5.RELAESE----->Reference Documentation学习心得----->Spring Framework的依赖注入和控制反转
Dependency Injection and Inversion of Control 1.概述: 1.1相关概念 bean:由IoC容器所管理的对象,也即各个类实例化所得对象都叫做bean 控制 ...
- 简单解析依赖注入(控制反转)在Spring中的应用
IoC——Inversion of Control 控制反转DI——Dependency Injection 依赖注入 大家都知道,依赖注入是Spring中非常重要的一种设计模式.可能很多初学者 ...
随机推荐
- PHP编程基础学习(一)——数据类型
PHP一共支持8种原始类型,其中: 4种标量类型: boolean(布尔型) integer(整型) float/double(浮点型) string(字符串型) 两种复合类型: array(数组) ...
- 你的跑步姿势正确吗? 教你正确跑步姿势 & 常识
转载!!!!!搞IT必须运动一下 前言: 最近两年跑步的人越来越多,跑步在大部分人的观念中都是毫无技术含量,只要迈开腿就行了,其实这也是造成大多数跑步人士伤病的根源.对跑步的认知不足,跑步是一项看起来 ...
- Django配置参数可选总结
一.可选字段参数 null blank core db_index editable primary_key radio_admin unique True or False db_colum hel ...
- django信号调度的用法
Django中提供了"信号调度",用于在框架执行操作时解耦. 一些动作发生的时候,系统会根据信号定义的函数执行相应的操作 Django中内置的signal Model_signal ...
- BZOJ 2333: [SCOI2011]棘手的操作 可并堆 左偏树 set
https://www.lydsy.com/JudgeOnline/problem.php?id=2333 需要两个结构分别维护每个连通块的最大值和所有连通块最大值中的最大值,可以用两个可并堆实现,也 ...
- 【spfa】【动态规划】zoj3847 Collect Chars
转载自:http://blog.csdn.net/madaidao/article/details/42616743 Collect Chars Time Limit: 2 Seconds ...
- 拆分Cocos2dx渲染部分代码
纹理实现 思想 这个是Cocos2dx的渲染部分的最基本的实现,被我拆分到mac上,但是并不是用的EGLContext,而是搭配glfw,还有soil第三方图形库. 实现 // // main.cpp ...
- 【原】不定义Order属性,通过切面类的定义顺序来决定通知执行的先后顺序
[结论] 在多个切面类的“切入点相同”并且每个切面都“没有定义order属性”的情况下,则切面类(中的通知)的执行顺序与该切面类在<aop:config>元素中“声明的顺序”相关,即先声明 ...
- css3实现卷页效果http://jingyan.baidu.com/article/73c3ce2806aef9e50343d93a.html
css3实现卷页效果 | 浏览:31 | 更新:2015-01-08 13:30 1 2 3 4 5 6 7 分步阅读 百度经验:jingyan.baidu.com 页面上经常会看到鼠标移动上去,对象 ...
- Android Tasker应用之自动查询并显示话费流量套餐信息
Android Tasker应用之自动查询并显示话费流量套餐信息 虽然Android平台有非常多的流量监控软件,但最准确的流量数据还是掌握在运营商手里.有些朋友可能像我一样时不时地发短信查询流量信息, ...