Spring 依赖注入(控制反转)介绍

耦合性是软件工程中的一个重要概念。对象之间的耦合性就是对象之间的依赖性。对象之间的耦合越高,维护成本越高。因此对象的设计应使类和构件之间的耦合最小。 spring Ioc思想 控制翻转也就是spring所推出的依赖注入

传统的方法假设我们前端用的struts,那么action中我们对业务逻辑的编写一般是这样的

这里是一段伪代码:

复制代码
publicclass CustomerDao {  publicboolean getCustomerByAccount() {/* 代码 */ }      } 
publicclass LoginAction {  publicvoid execute() {              CustomerDao cdao =new CustomerDao();  boolean b = cdao.getCustomerByAccount();  // 判断          }      }
复制代码

在LoginAction中,execute内直接实例化了CustomerDao对象,此时,相当于让LoginAction的使用依赖了 CustomerDao,换句话说,如果没有CustomerDao类,LoginAction将无法被编译、测试。另外,如果CustomerDao类 换成另一版本,比如原先的CustomerDao,是访问SQLServer数据库,后面我们又另外编写了CustomerDao2类,访问Oracle 数据库,那么需要将LoginAction内所有出现CustomerDao的地方改为CustomerDao2。非常麻烦。这就是耦合性高的代价。

那么如何降低耦合性呢,我们可以将变化较大的类改成变化较小的接口,然后在action中通过一个工厂类来返回相应的对象。将代码修改为:

复制代码
publicinterface ICustomerDao {  publicboolean getCustomerByAccount();      } 
publicclass CustomerDao implements ICustomerDao {  publicboolean getCustomerByAccount(){/* 代码 */ }      }  //工厂类publicclass DaoFactory {  publicstatic ICustomerDao getCustomerDao() {  returnnew CustomerDao();          }      } 
publicclass LoginAction {  publicvoid execute() {              ICustomerDao icdao = DaoFactory.getCustomerDao();  boolean b = icdao.getCustomerByAccount();  // 判断          }      }
复制代码

在上面的程序中,如果需要做CustomerDao的切换,如,从CustomerDao改为CustomerDao2,就只需让CustomerDao2实现"ICustomerDao"接口,然后修改工厂的方法,而不用修改LoginAction内的代码。

LoginAction只需要认识ICustomerDao接口,不需要认识具体的实现类。而接口修改的概率比实现类要低得多。因此,这样编写就降低了程序的耦合性,是一个比较好的方法。

但是,以上方法也不是没有修改的余地。当Dao进行切换时,还是需要修改BeanFactory的源代码,能否避免这个问题呢?可以对DaoFactory进行改进,使得它能为所有类服务。代码如下

复制代码
publicinterface ICustomerDao {  publicboolean getCustomerByAccount();      } 
publicclass CustomerDao implements ICustomerDao {  publicboolean getCustomerByAccount() {/* 代码 */          }      } 
publicclass BeanFactory {  publicstatic Object getBean(String className) {  return Class.forName(className).newInstance();          }      } 
publicclass LoginAction {  publicvoid execute() {              ICustomerDao icdao = (ICustomerDao)BeanFactory.getBean("CustomerDao");  boolean b = icdao.getCustomerByAccount();  // 判断          }      }
复制代码

此处使用了反射机制。修改后,需要切换的时候只需在LoginAction内改变类名,工厂内就会自动生成对象返回给LoginAction。在 LoginAction中,由于类名是字符串,因此,就可以将该字符串写在一个配置文件内,让LoginAction读入,这样,当 CustomerDao类名需要切换时,就直接修改配置文件就行了。不用改源代码,模块之间的耦合就完全由配置文件决定。

这种设计方法有一个好处是,BeanFactory类的通用性很强,就可以将其框架化。因此,框架化之后,对象的生成由框架参考配置文件进行,和具 体实现类的源代码无关,将对象生成的控制权由修改不方便的源代码转变为修改相对方便的配置文件与几乎不进行修改的框架进行,这就是控制反转 (Inverse Of Control,IOC)的原理。

原文地址:https://www.cnblogs.com/heartstage/p/3390430.html