Spring IoC
在前两篇文章中,我们讲了java web环境搭建 和 java web项目搭建,现在看下spring ioc在java中的运用,开发工具为Intellij Idea。
1.简介
IoC(Inversion of Control
)通常称为控制反转,是Spring框架的核心。即通过Spring容器控制对象之间的依赖关系,而不是对象自己通过new的方式去创建依赖的对象,相对于主动去创建对象,这种失去对对象的控制便叫做控制反转,控制反转用另一种角度描述或者更好的一种理解方式来讲,便是DI(Dependency Injection
)依赖注入,对象只需关注业务逻辑本身即可,而不需关注依赖的对象从哪里来,交给容器注入即可,从而降低程序之间的耦合度。
2.用法
依赖注入在spring中有两种配置方式,一是xml配置Bean的方式,二是自动装配检测Bean
-
2.1 xml配置Bean
- 2.1.1 创建Spring配置
我们在web层,web-inf/dispatcher-servlet.xml
中注入service层的实现类,加入如下代码
<bean id="demoUserService" class="com.ganji.demo.service.user.DemoUserServiceImpl" />
- 2.1.2 在web层controller下调用
我们在类里声明对象,并在方法里调用,这里我们的service层接口类可以声明为私有属性,程序可通过@Inject
自动搜索xml中的bean配置,注入依赖。
// 声明对象 DemoUserServiceImpl实现DemoUserService接口 @Inject private DemoUserService demoUserService; @RequestMapping(value="/index", method = {RequestMethod.GET}) public ModelAndView index(){ DemoUserEntity demoUser=demoUserService.GetDemoUser(1); modelAndView.addObject("demoUser", demoUser); modelAndView.setViewName("home"); return modelAndView; }
我们在属性前加了
@Inject
,这里依赖javax.inject.Inject
包 ,在模块pom里加入如下依赖即可<dependency> <groupId>javax.inject</groupId> <artifactId>javax.inject</artifactId> </dependency>
- 2.1.3 属性的依赖注入
如果我们想在
service
层依赖注入调用dao层对象操作数据库,则跟service层类似,我们在web层的web-inf/dispatcher-servlet.xml
注入dao
层的实现,加入如下代码<bean id="demoUserDao" class="com.ganji.demo.dao.gcrm.hibernate.DemoUserDaoHibernate"></bean> <bean id="demoUserService" class="com.ganji.demo.service.user.DemoUserServiceImpl" autowire="byType"> <property name="demoUserDao" ref="demoUserDao"></property> </bean>
ref
指像bean
的实现类,如果是值类型或String
类型可以用value
指定,设定值即可,如value=5
。
在service
层,我们可以在DemoUserServiceImpl
层里声明demoUserDao
私有属性,并公开属性set方法,然后调用//声明属性 private DemoUserDao demoUserDao; //通过属性的set方法,注入对象 public void setDemoUserDao(DemoUserDao demoUserDao) { this.demoUserDao=demoUserDao; } public DemoUserEntity GetDemoUser(int id) { return demoUserDao.getDemoUser(id); }
- 2.1.4 构造器注入
构造器注入类似于属性注入,在xml中用
constructor-arg
来指定,这里我们在web层的web-inf/dispatcher-servlet.xml
配置如下<bean id="demoUserService" class="com.ganji.demo.service.user.DemoUserServiceImpl" autowire="byType"> <constructor-arg name="demoUserDao" ref="demoUserDao"></constructor-arg> <constructor-arg name="userName" value="张三"></constructor-arg> </bean>
在service层,我们创建构造函数
private DemoUserDao demoUserDao; private String userName; public DemoUserServiceImpl(DemoUserDao demoUserDao,String userName) { this.demoUserDao=demoUserDao; this.userName=userName; } //在方法里既可以调用了,如下代码 public DemoUserEntity GetDemoUser(int id) { System.out.println("execute service getdemouser "+userName); return demoUserDao.getDemoUser(id); }
- 2.1.5 小结
如果我们不想在xml中配置属性注入或者构造器注入,我们可以在声明的属性上,添加@Inject
注解,类似2.1.2中demoUserService的实现,这样Spring框架会自动搜索bean对应的实现类,可以在bean处设置根据名称或类型,即autowire="byType" or autowire="byName"
,也可以全局设置,即在根目录beans下面声明default-autowire="byName"
,具体如下。
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation=" http://www.springframework.org/schema/beans" default-autowire="byName" >
- 2.1.1 创建Spring配置
-
2.2自动装配检测Bean
如果每个实现类都要配置bean元素的话,项目如果很大的情况下,太多的bean,xml文件的bean管理配置就成了一个问题,很庆幸的事,spring提供了自动装配检测bean注入依赖的功能。
- 2.2.1 自动检测标注bean
我们依赖
<context:component-scan>
查找使用注解所标注的类,这些注解如下@Component --通用的构造性注解,标识该类为Spring组件
@Controller --标识将该类定义为Spring MVC Controller 这里我们用在web层
@Service --标识将该类定义为服务,这里我们用在Service层
@Repository --标识将该类定义为数据仓库,这里我们用在Dao层具体用法如下图示例
web层调用
@Controller
public class HelloWorldController {service层调用
@Service
public class DemoUserServiceImpl implements DemoUserService {dao层调用
@Repository
public class DemoUserDaoHibernate implements DemoUserDao {注解后,我们需要在web层的
web-inf/dispatcher-servlet.xml
配置组件扫描<context:component-scan>
,分别扫描web、service。dao层,具体如下<!-- 开启controller注解支持 --> <!-- use-default-filters="false" 只扫描指定的注解 --> <context:component-scan base-package="com.ganji.demo.web.controller" use-default-filters="false"> <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller" /> </context:component-scan> <!--服务层自动依赖注入--> <context:component-scan base-package="com.ganji.demo.service"> <context:include-filter type="annotation" expression="org.springframework.stereotype.Service" /> </context:component-scan> <!--Dao层依赖注入--> <context:component-scan base-package="com.ganji.demo.dao"> <context:include-filter type="annotation" expression="org.springframework.stereotype.Repository" /> </context:component-scan>
同时,我们需要在xml命名空间里加入context支持,
xmlns:context="http://www.springframework.org/schema/context"
和相应的xsi具体如下<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd" default-autowire="byName" >
- 2.2.2 使用自动注入的bean
上一步配置好后,我们只需要声明调用对象的属性即可,如下
web层如下// 声明对象 DemoUserServiceImpl实现DemoUserService接口 @Inject private DemoUserService demoUserService; @RequestMapping(value="/index", method = {RequestMethod.GET}) public ModelAndView index(){ DemoUserEntity demoUser=demoUserService.GetDemoUser(1); modelAndView.addObject("demoUser", demoUser); modelAndView.setViewName("home"); return modelAndView; }
service层如下
@Inject private DemoUserDao demoUserDao; public DemoUserEntity GetDemoUser(int id) { return demoUserDao.getDemoUser(id); }
- 2.2.3 总结
按2.2.1和2.2.2配置操作后,即可实现自动检查依赖注入bean,不用在xml中配置繁琐的bean元素,由框架通过反射自动寻找对象实现。