SpringMVC实现依赖注入的几个常用注解的用法解释和示例说明

简介:

在以前,我们使用Spring对一些属性进行依赖注入(DI)时,通常都是通过在配置文件中配置一个个的<bean>标签来实现,比如说这样:

1
2
3
< bean  id = "userBiz"  class = "cn.zifangsky.biz.UserBizImpl" >
     < property  name = "userDAO"  ref = "userDAO" />
</ bean >

但是一旦项目大了之后,如果要把所有的这些依赖关系都在配置文件中配置的话,无疑逻辑上是非常混乱的,这时我们就可以考虑使用几个常用的注解来实现依赖关系的注入。

实现依赖注入的几个常用注解分别是:

  • @Component 是所有受Spring 管理组件的通用形式,但是不推荐使用

  • @Repository 对应数据访问层的Bean 

  • @Service 对应业务层的Bean

  • @Controller 对应控制层的Bean

除了这几个对类的注解外,还有几个对类中属性的注解,主要目的是告诉Spring这个属性应该用那个前面已经注解过的类来实例化,它们分别是:

  • @Resource 默认按名称来装配注入,只有当找不到与名称相匹配的bean时才会按照类型来装配注入

  • @Autowired 默认按类型来装配注入,如果想按照名称来装配注入,则需要结合@Qualifier一起使用

  • @Qualifier 默认按名称来装配注入

由于@Autowired是按照类型来注入的,因此当同类型的变量有多个需要注入时,仅仅使用@Autowired就会出现问题,这时可以结合@Qualifier来使用,比如:

1
2
3
@Autowired
@Qualifier ( "userDaoImpl" )
private  UserDao userDao;

当然,由于@Resource这个注解有另外两个注解都有的功能,同时@Resource这个注解是基于J2EE的,而@Autowired和@Qualifier是属于Spring的,所以我们最好使用@Resource进行依赖注入,有利于减小应用与Spring的耦合

对于上面提到的这些注解的一些具体用法,接下来我将以一个具体的实例来举例说明:

一 测试环境的搭建

1 项目结构与相关jar包:

wKioL1em8ZTi9OnIAABbQjJYNiU340.png  wKiom1em8Z_ARaAmAAAaRZqXDBU345.png

2 web.xml以及springmvc-servlet.xml:

i)web.xml:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
< web-app  xmlns = "http://xmlns.jcp.org/xml/ns/javaee"
          xmlns:xsi = "http://www.w3.org/2001/XMLSchema-instance"
          xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee
       http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
          version = "3.1" >
 
     < servlet >
         < servlet-name >springmvc</ servlet-name >
         < servlet-class >org.springframework.web.servlet.DispatcherServlet</ servlet-class >
         < load-on-startup >1</ load-on-startup >
     </ servlet >
 
     < servlet-mapping >
         < servlet-name >springmvc</ servlet-name >
         < url-pattern >*.html</ url-pattern >
     </ servlet-mapping >
     
     < filter >
         < filter-name >characterEncodingFilter</ filter-name >
         < filter-class >org.springframework.web.filter.CharacterEncodingFilter</ filter-class >
         < init-param >
             < param-name >encoding</ param-name >
             < param-value >UTF-8</ param-value >
         </ init-param >
     </ filter >
     < filter-mapping >
         < filter-name >characterEncodingFilter</ filter-name >
         < url-pattern >/*</ url-pattern >
     </ filter-mapping >
 
</ web-app >

ii)springmvc-servlet.xml:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
<? 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"
     xmlns:mvc = "http://www.springframework.org/schema/mvc"
     xsi:schemaLocation="http://www.springframework.org/schema/beans 
                             http://www.springframework.org/schema/beans/spring-beans-4.0.xsd 
                             http://www.springframework.org/schema/context
                             http://www.springframework.org/schema/context/spring-context-4.0.xsd
                             http://www.springframework.org/schema/mvc 
                             http://www.springframework.org/schema/mvc/spring-mvc-4.0.xsd">
 
     < context:component-scan  base-package = "cn.zifangsky.dao"  />
     < context:component-scan  base-package = "cn.zifangsky.service"  />
     < context:component-scan  base-package = "cn.zifangsky.controller"  />
     <!-- <context:component-scan base-package="cn.zifangsky.* *.controller" /> -->
 
     < context:annotation-config  />   <!-- 激活Bean中定义的注解 -->
     < mvc:annotation-driven  /> 
 
     < bean
         class = "org.springframework.web.servlet.view.InternalResourceViewResolver" >
         < property  name = "prefix"  value = "/WEB-INF/pages/"  />
         < property  name = "suffix"  value = ".jsp"  />
     </ bean >
</ beans >

可以看出,在上面的配置文件中定义了三个“context:component-scan”标签分别用于扫描数据访问层、业务层以及控制层的注解,其他配置不用多说

二 注解的使用实例

1 首先是model层的User.java:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
package  cn.zifangsky.model;
 
public  class  User {
     private  String name;
     private  String password;
 
     public  String getName() {
         return  name;
     }
 
     public  void  setName(String name) {
         this .name = name;
     }
 
     public  String getPassword() {
         return  password;
     }
 
     public  void  setPassword(String password) {
         this .password = password;
     }
 
}

这一层只是简单定义了一个“User”模型,并没有使用到任何注解

2 数据访问层的UserDao.java和UserDaoImpl.java:

i)UserDao.java:

1
2
3
4
5
6
7
8
9
10
package  cn.zifangsky.dao;
 
import  cn.zifangsky.model.User;
 
public  interface  UserDao {
     /**
      * 校验登录信息
      * */
     public  String check(User u);
}

ii)UserDaoImpl.java:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
package  cn.zifangsky.dao.impl;
 
import  org.springframework.stereotype.Repository;
 
import  cn.zifangsky.dao.UserDao;
import  cn.zifangsky.model.User;
 
@Repository ( "userDaoImpl" )
public  class  UserDaoImpl  implements  UserDao {
 
     public  String check(User u) {
         //这里只是简单判断是否为空,实际需要查询数据库等操作
         if (u !=  null ){
             System.out.println( "登录的用户信息是:" );
             System.out.println( "--|用户名:"  + u.getName());
             System.out.println( "--|密码:"  + u.getPassword());
         }
 
         return  "ok" ;
     }
 
}

可以看出,在上面的代码中我们在类上定义了一个@Repository注解,用于表示这个类是数据访问层的,同时给它起了个名字叫“userDaoImpl”。问:为什么在实现类上添加@Repository注解而不是在UserDao接口上添加注解?

其实这个问题只要想一下我们在配置文件中使用<bean>标签是如何配置的就清楚了,比如对于这样一个bean:

1
2
3
< bean  id = "userDAO"  class = "cn.zifangsky.dao.UserDAOImpl" >
     < property  name = "sessionFactory"  ref = "sessionFactory"  />
</ bean >

我们可以发现,“class”属性这里指向的是一个具体的实现类而不是它的接口。原因很简单,我们需要的是一个具体的类来实例化或者说在其他类中也需要这样一个具体的类来进行参数注入,显然这是不能使用接口的。因此,上面为何在一个实现类上添加注解也是基于同样的道理,不信可以试试把@Repository注解从UserDaoImpl转移到UserDao这个接口上,看看项目运行时会不会报错?

3 业务逻辑层UserService.java和UserServiceImpl.java:

i)UserService.java:

1
2
3
4
5
6
7
8
9
10
package  cn.zifangsky.service;
 
import  cn.zifangsky.model.User;
 
public  interface  UserService {
     /**
      * 模拟登录
      * */
     public  String login(User u);
}

ii)UserServiceImpl.java:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
package  cn.zifangsky.service.impl;
 
import  javax.annotation.Resource;
 
import  org.springframework.stereotype.Service;
 
import  cn.zifangsky.dao.UserDao;
import  cn.zifangsky.model.User;
import  cn.zifangsky.service.UserService;
 
@Service ( "userServiceImpl" )
public  class  UserServiceImpl  implements  UserService {
     @Resource (name =  "userDaoImpl" )
     private  UserDao userDao;
 
//  public void setUserDao(UserDao userDao) {
//      this.userDao = userDao;
//  }
 
     public  String login(User u) {
         return  userDao.check(u);
     }
}

在这里,UserServiceImpl这个类上面定义了一个@Service注解,表示它是业务逻辑层上的一个类,同样给它起了一个名字叫“userServiceImpl”。通过上面的代码可以看到,我们给userDao这个属性定义了一个@Resource注解,通过一个“name”属性表示引用的是一个名为“userDaoImpl”的UserDao类型的类来实例化userDao属性,毫无疑问这里指的就是上面定义了“@Repository(“userDaoImpl”)”注解的UserDaoImpl.java了

注:给属性添加了注解之后是可以不用再写对应的setter方法的

4 控制层UserController.java:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
package  cn.zifangsky.controller;
 
import  javax.annotation.Resource;
 
import  org.springframework.stereotype.Controller;
import  org.springframework.web.bind.annotation.RequestMapping;
import  org.springframework.web.bind.annotation.RequestParam;
 
import  cn.zifangsky.model.User;
import  cn.zifangsky.service.UserService;
 
@Controller
public  class  UserController {
     @Resource (name= "userServiceImpl" )
     private  UserService userService;
 
     @RequestMapping (value= "/analogLogin" )
     public  String user( @RequestParam (name= "name" ,required= false ) String name, @RequestParam (name= "password" ,required= false ) String password){
         User user =  new  User();
         user.setName(name);
         user.setPassword(password);
         
         String result = userService.login(user);
         System.out.println(result);
     
         return  "success" ;
     }
     
}

@Controller注解的含义这里就不用多说了,跟上面的@Repository和@Service类似。通过定义的@RequestMapping注解我们在访问“http://localhost:[port]/AnnotationDemo/analogLogin.html”就可以转到user这个方法进行处理。而@RequestParam注解则是用来获取请求中的参数用的

三 测试

在浏览器中访问:http://localhost:8090/AnnotationDemo/analogLogin.html?name=haha&password=123qwe

对应的输出如下:

1
2
3
4
登录的用户信息是:
--|用户名:haha
--|密码:123qwe
ok

如果出现上面的控制台输出则表明注解已经配置正确了



本文转自 pangfc 51CTO博客,原文链接:http://blog.51cto.com/983836259/1835371,如需转载请自行联系原作者

相关文章
|
4月前
|
前端开发 Java API
Spring Boot之Spring MVC基于注解的控制器(RequestMapping注解类型 重定向与转发 依赖注入)
Spring Boot之Spring MVC基于注解的控制器(RequestMapping注解类型 重定向与转发 依赖注入)
45 0
|
4月前
|
设计模式 前端开发 JavaScript
Spring MVC(一)【什么是Spring MVC】
Spring MVC(一)【什么是Spring MVC】
|
1月前
|
Java Apache vr&ar
springmvc报错 nested exception is org.mybatis.spring.MyBatisSystemException:
springmvc报错 nested exception is org.mybatis.spring.MyBatisSystemException:
16 0
|
6月前
|
前端开发 Java Go
Spring MVC 和 Spring Boot 的区别
Spring MVC 和 Spring Boot 的区别
110 0
|
9月前
|
Java Spring
springmvc中spring提供的中文乱码解决问题
可以解决浏览器的乱码问题
44 0
|
4月前
|
设计模式 前端开发 Java
Spring Boot之Spring MVC的工作原理 以及使用eclipse开发Spring MVC的Web应用实战(附源码)
Spring Boot之Spring MVC的工作原理 以及使用eclipse开发Spring MVC的Web应用实战(附源码)
45 0
|
4月前
|
设计模式 前端开发 Java
8:Spring MVC-Java Spring
8:Spring MVC-Java Spring
72 0
|
9月前
|
JSON 前端开发 Java
"《图书管理系统》利用SpringMvc$Spring$MyBatis (实操九)(一) "
"《图书管理系统》利用SpringMvc$Spring$MyBatis (实操九)(一) "
57 0
|
8月前
|
前端开发 Java 应用服务中间件
SpringMVC(基于Spring 的Web 层MVC 框架)--SpingMVC 执行流程--@RequestMapping的使用
SpringMVC(基于Spring 的Web 层MVC 框架)--SpingMVC 执行流程--@RequestMapping的使用
97 0
|
9月前
|
SQL JSON 前端开发
SpringMvc+Spring+MyBatis+Maven+Ajax+Json注解开发 利用Maven的依赖导入不使用架包模式 (实操十二)
SpringMvc+Spring+MyBatis+Maven+Ajax+Json注解开发 利用Maven的依赖导入不使用架包模式 (实操十二)
44 0