专栏首页Java大联盟对比Spring IoC两种开发方式:XML和注解

对比Spring IoC两种开发方式:XML和注解

今天楠哥给大家讲讲在实际开发中经常会使用到的 IoC 技术:通过 IoC 容器架构程序的分层实现,有两种方式:基于 XML 配置文件和基于注解。

我们把程序分为 3 层:Controller 层、Service 层和 DAO 层。关系为 Controller 层调用 Service 层,Service 层调用 DAO 层,并且 Service 层和 DAO 层设计为接口,这是一个典型的 MVC 模式后台代码分层结构。

基于 XML 配置方式

(1)创建 UserController 类:

public class UserController {

     private UserService userService;

     public User getUserById(int id){
         return userService.getUserById(id);
     }

}

(2)创建 UserService 接口以及实现类 UserServiceImpl:

public interface UserService {
     public User getUserById(int id);
}

public class UserServiceImpl implements UserService{

     private UserDAO userDAO;

     @Override
     public User getUserById(int id) {
         // TODO Auto-generated method stub
         return userDAO.getUserById(id);
     }

}

(3)创建 UserDAO 接口以及实现类 UserDAOImpl:

public interface UserDAO {
     public User getUserById(int id);
}

public class UserDAOImpl implements UserDAO{

     private static Map<Integer,User> users;

     static{
         users = new HashMap<Integer,User>();
         users.put(1, new User(1, "张三"));
         users.put(2, new User(2, "李四"));
         users.put(3, new User(3, "王五"));
     }

     @Override
     public User getUserById(int id) {
         // TODO Auto-generated method stub
         return users.get(id);
     }

}

(4)创建 User 实体类:

public class User {
     private int id;
     private String name;
     public User(int id, String name) {
         super();
         this.id = id;
         this.name = name;
     }
}

(5)在 spring.xml 中配置

userController、userService、userDAO,并完成依赖注入:

<!-- 配置 UserController -->
<bean id="userController" class="com.southwind.controller.UserController">
 <property name="userService" ref="userService"></property>
</bean>
<!-- 配置 UserService -->
<bean id="userService" class="com.southwind.service.impl.UserServiceImpl">
 <property name="userDAO" ref="userDAO"></property>
</bean>
<!-- 配置 UserDAO -->
<bean id="userDAO" class="com.southwind.dao.impl.UserDAOImpl"></bean>

(6)在测试类中获取 userController 对象,调用方法获取 user 对象:

ApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring.xml");
UserController userController = (UserController) applicationContext.getBean("userController");
User user = userController.getUserById(1);
System.out.println(user);

基于注解的方式

将 UserController、UserService、UserDAO 类扫描到 IoC 容器中,在类中设置注解完成依赖注入。

(1)修改 spring.xml:

<!-- 将类扫描到 IoC 容器中 -->
<context:component-scan base-package="com.southwind"></context:component-scan>

base-package="com.southwind" 表示将 com.southwind 下所有子包的类全部扫描到 IoC 容器中,一步可将所有参与项目的类完成扫描注入。注意:需要引入 context 命名空间。

(2)修改 UserController,添加注解:

@Controller
public class UserController {

     @Autowired
     private UserService userService;

     public User getUserById(int id){
         return userService.getUserById(id);
     }

}

对比之前的代码,有两处改动:

1、在类名处添加 @Controller 注解,表示该类作为一个控制器;

2、userService 属性处添加 @Autowired 注解,表示 IoC 容器自动完成装载,默认是 byType 的方式。

(3)修改 UserServiceImpl:

@Service
public class UserServiceImpl implements UserService{

     @Autowired
     private UserDAO userDAO;

     @Override
     public User getUserById(int id) {
         return userDAO.getUserById(id);
     }

}

同上,做了两处改动:

1、在类名处添加 @Service 注解,表示该类是业务层;

2、userDAO 属性处添加 @Autowired 注解,表示 IoC 容器自动完成装载,默认是 byType 的方式。

(4)修改 UserDAOImpl:

@Repository
public class UserDAOImpl implements UserDAO{

     private static Map<Integer,User> users;

     static{
         users = new HashMap<Integer,User>();
         users.put(1, new User(1, "张三"));
         users.put(2, new User(2, "李四"));
         users.put(3, new User(3, "王五"));
     }

     @Override
     public User getUserById(int id) {
         // TODO Auto-generated method stub
         return users.get(id);
     }

}

做了一处改动:在类名处添加 @Repository 注解,表示该类是数据接口层。

(5)运行测试代码:

ApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring.xml");
UserController userController = (UserController) applicationContext.getBean("userController");
User user = userController.getUserById(1);
System.out.println(user);

代码成功运行,通过代码可以看到,使用注解的方式,可简化代码,所以实际开发中,推荐使用基于注解的方式来架构分层。

我们分别给 UserController、UserService、UserDAO 添加了 @Controller、@Service、@Repository 注解。

IoC 中可以给类添加的注解有 4 种:

@Controller

@Service

@Repository

@Component

这 4 种注解方式没有区别,我们在开发时可以随意使用任意一个注解,但是基于代码规范一般选择使用 @Controller、@Service、@Repository 分别表示 Controller 层、Service 层、DAO 层。

前面我们提到过,类中属性自动装载,默认是通过 byType 的方式,自动装载除了 byType 的方式,还可以使用 byName 的方式,使用 byName 的方式,需要结合 @Qualifier 注解一起使用。

@Controller
public class UserController {

     @Autowired()
     @Qualifier("userService")
     private UserService userService;

     public User getUserById(int id){
         return userService.getUserById(id);
     }
}

我们知道 byName 的方式,是通过属性名去匹配对应 bean 的 id 属性值,但是基于注解的方式我们并没有给 bean 设置 ID,该如何完成呢?

其实在类中添加注解时,已经设置了默认的 ID,即类名首字母小写之后的值就是 ID 的默认值。

@Service
public class UserServiceImpl implements UserService

此时,IoC 容器中默认赋值,UserService bean 的id=userService,与 UserController 中的属性名一致,所以可以完成自动。

现在做出修改手动赋值,设置UserService bean 的 id=myUserService。

@Service("myUserService")
public class UserServiceImpl implements UserService{

     @Autowired
     private UserDAO userDAO;

     @Override
     public User getUserById(int id) {
         // TODO Auto-generated method stub
         return userDAO.getUserById(id);
     }

}

很显然,UserController 中的 userService 属性也需要去匹配 name=myUserService 的bean,所以设置 @Qualifier("myUserService")。

@Controller
public class UserController {

     @Autowired()
     @Qualifier("myUserService")
     private UserService userService;

     public User getUserById(int id){
         return userService.getUserById(id);
     }
}

@Qualifier() 中的值必须与 @Service() 中的值一致,才能完成自动装载。

推荐阅读

1、axios异步请求数据的12种操作(上篇)

2、axios异步请求数据的12种操作(下篇)

3、一文搞懂前后端分离

4、快速上手Spring Boot+Vue前后端分离

本文分享自微信公众号 - Java大联盟(javaunion),作者:楠哥

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2019-11-13

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • Java基础之面向对象

    南风
  • Java基础之面向对象

    在Java中,有句俗语:万物皆对象。我们常常讲Java是一门面向对象的编程语言,那到底什么是“面向对象”、除了“面向对象”还有哪种编程思想、为什么要采用“面向对...

    南风
  • 没时间直播,先来一波 Shiro 整合 Spring Boot 图文教程!

    Apache Shiro 是一款 Java 安全框架,不依赖任何容器,可以运行在 Java SE 和 Java EE 项目中,它的主要作用是用来做身份认证、授权...

    南风
  • 聊聊flink DataStream的iterate操作

    flink-streaming-java_2.11-1.7.0-sources.jar!/org/apache/flink/streaming/api/data...

    codecraft
  • 【Android】造轮子:轮播图

    Gavin-ZYX
  • 一个Java能设置key过期时间的HashMap

    神秘的寇先森
  • C# 基础知识系列- 14 IO篇之入门IO

    在之前的章节中,大致介绍了C#中的一些基本概念。这篇我们将介绍一下C#的I/O操作,这将也是一个小连续剧。这是第一集,我们先来简单了解一下C#中的I/O框架。

    程序员小高
  • SpringBoot应用篇之FactoryBean及代理实现SPI机制示例

    FactoryBean在Spring中算是一个比较有意思的存在了,虽然在日常的业务开发中,基本上不怎么会用到,但在某些场景下,如果用得好,却可以实现很多有意思的...

    一灰灰blog
  • 【小家Spring】Spring MVC容器的web九大组件之---HandlerAdapter源码详解---HttpMessageConverter 消息转换器详解

    本文介绍Spring MVC中的一个极其重要的组件:HttpMessageConverter消息转换器。

    YourBatman
  • cf19E. Fairy(奇环 二分图染色)

    非常有思维含量的一道题,队爷的论文里介绍了一种\(N \sqrt{N}\)的暴力然鹅看不懂。。

    attack

扫码关注云+社区

领取腾讯云代金券