Spring的包扫描开发与@Autowired与@Resource注解的区别(五)

简介: Spring的包扫描开发与@Autowired与@Resource注解的区别(五)

一.注解开发的意义


前面第四章所用的是XML配置的方式进行的开发,在实际的开发中,Action有多个,所对应的Service与Dao也有很多个,会导致存在很多的bean的情况。 而且,特别是简单的实体类POJO 会有很多的属性,会导致要写很多的<property> 进行注入,这样非常的不好,可以采用注解的方式进行相应的配置,可以简化开发。


例子与第四章的例子是一样的。


所使用的jar包有:


20190417194717702.png


其中,spring-aop 与struts2-spring-plugin 的jar包是这次要用到的。其余的,都是上次第四章用到的。


二. 注解开发


添加注解,需要用到 aop的jar包。


二.一 添加aop 的jar 包。


20190420161535856.png


二.二 在applicationContext.xml 中添加约束


添加的约束为 context。


添加后的约束为:


<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">


二.三 添加组件扫描


其中,base-package 指定要扫描的包名, 如果有多个的话,之间用,进行隔开。


这个包名要尽量小,这样就可以少扫描一些无用的包,但要把


 <!-- 组件扫描 -->
  <context:component-scan base-package="com.yjl"></context:component-scan>
<!--多个的话-->
<context:component-scan base-package="com.yjl,com.abc,com.bcd"></context:component-scan>


二.四 在service中添加dao的引用


@Service
public class UserServiceImpl implements UserService{
  @Autowired
  private UserDao userDao;
  @Override
  public List<User> listAll() {
    System.out.println("输出userDao:"+userDao);
    return userDao.listAll();
  }
}


在UserServiceImpl 上,添加了一个@Service 的类注解, 在userDao 属性上面添加了一个@Autowired 的注解。 这里并没有实现userDao的setter和getter 方法。


二.五 UserDaoImpl


@Repository
public class UserDaoImpl implements UserDao {
  @Override
  public List<User> listAll() {
    List<User> userList=new ArrayList<User>();
    User user1=new User(1,"父亲","男",50);
    User user2=new User(2,"母亲","女",48);
    User user3=new User(3,"慧芳","女",27);
    User user4=new User(4,"正伟","男",28);
    User user5=new User(5,"莉莉","女",25);
    User user6=new User(6,"敬龙","男",26);
    User user7=new User(7,"两个蝴蝶飞","男",24);
    userList.add(user1);
    userList.add(user2);
    userList.add(user3);
    userList.add(user4);
    userList.add(user5);
    userList.add(user6);
    userList.add(user7);
    return userList;
  }
}


在UserDaoImpl 上添加了 @Repository 的注解, listAll() 的方法改成了普通的方法。


二.六 在Action 中注入service


@Controller
public class UserAction extends ActionSupport{
  private static final long serialVersionUID = 1L;
  @Autowired
  private UserService userService;
  /**
   * @author yuejl
   * @Description 查询全部
   * @return
   */
  public String list(){
    System.out.println("输出userService:"+userService);
    List<User> userList=userService.listAll();
    ServletActionContext.getRequest().setAttribute("userList",userList);
    return "list";
  }
  public UserService getUserService() {
    return userService;
  }
  public void setUserService(UserService userService) {
    this.userService = userService;
  }
}


在UserAction 上面有一个@Controller 的类注解, userService 属性上有一个@Autowired 的属性注解。


二.七 struts.xml 文件


<!--注意,此时class仍然为全限定名称-->
<action name="User_*" class="com.yjl.web.UserAction" method="{1}">
      <result name="list">list.jsp</result>
    </action>


二.八 重启服务器,看是否成功


20190420163445967.png


20190420163502755.png


可以正常的进行注入。


二.九 将action中@Autowired 注入到setter方法上


2019042016360648.png


也是可以正常通过的。


20190420163703248.png


三. 类注解


类上的注解有四个, @Component, @Controller,@Service, @Repository


以前只有一个注解 @Component, 现在扩充了后面的三个。 目前来说,这四个没有区别,但以后可能会扩充后面三个的意义,所以一般:


@Controller 用于Web层

@Service 用于事务层

@Repository 用于持久层


四. @Autowired 注解


@AutoWired 是按照类型,即byType 的方式进行装配的,默认是必须要存在的。


如果不存在的话,即在Action中注入service时,而将 service的注解进行去除:


20190422185941678.png


这个时候,会报错的。 是在服务器启动的时候,就报错了。


Caused by: org.springframework.beans.factory.BeanCreationException: Could not autowire field:
 private com.yjl.service.UserService com.yjl.web.action.UserAction.userService; 
 nested exception is org.springframework.beans.factory.NoSuchBeanDefinitionException: 
 No qualifying bean of type [com.yjl.service.UserService] found for dependency: 
 expected at least 1 bean which qualifies as autowire candidate for this dependency.
 Dependency annotations: {@org.springframework.beans.factory.annotation.Autowired(required=true)}


可以手动设置 为非必须。


@Autowired(required=false)
  private UserService userService;


在访问 User_list.action 时会报错,是空指向的异常。 服务器启动的时候不报错。


20190422190257455.png


2. 是按照类型进行的自动装配,现在 UserService 接口只有一个实现 UserServiceImpl 类,所以按照类型找的话,只会找到这一个,可以自动进行装配。 如果现在UserService 多了一个实现类型,即UserServiceImpl2时,这个时候会怎么样呢?


@Service
public class UserServiceImpl2 implements UserService{
  @Autowired
  private UserDao userDao;
  @Override
  public List<User> listAll() {
    System.out.println("输出userDao:"+userDao);
    return userDao.listAll();
  }
}


那么这个Action的话:


@Autowired
  private UserService userService;


这个时候会产生异常:


Caused by: org.springframework.beans.factory.BeanCreationException: Could not autowire field: 
private com.yjl.service.UserService com.yjl.web.action.UserAction.userService;
 nested exception is org.springframework.beans.factory.NoUniqueBeanDefinitionException: 
 No qualifying bean of type [com.yjl.service.UserService] is defined: 
expected single matching bean but found 2: userServiceImpl,userServiceImpl2


这个时候,如果按照类型的 话, 会找到两个实现, userServiceImpl 和userServiceImpl2 时,这个时候是没有办法区分的装配的,会出现错误。


3. 解决,针对上面的问题,可以添加另外一个注解 @Qualifier ,这个来指定名称。


如: @Qualifier(value="userServiceImpl") 这个userServiceImpl 表示的就是UserServiceImpl, @Qualifier(value="userServiceImpl2") 表示的就是UserServiceImpl2,


其中,有一个对应的关系, value 的名称为类名称首字母小写。


解决方式: 添加一个@Qualifier 的注解。


五 @Resource 注解


@Resource 的注解来源包是: import javax.annotation.Resource; 是java 规范的,而@Autowired来源包是: import org.springframework.beans.factory.annotation.Autowired;

故最好使用 @Resouce 的注解。 使用@Resource 需要指定一下名称。


五.一 UserDaoImpl


/**
 @author:yuejl
 @date: 2019年4月20日 上午10:17:09
 @Description 类的相关描述
*/
// 需要添加一个 value的值
@Repository(value="userDaoImpl")
public class UserDaoImpl implements UserDao {
  @Override
  public List<User> listAll() {
    List<User> userList=new ArrayList<User>();
    User user1=new User(1,"父亲","男",50);
    User user2=new User(2,"母亲","女",48);
    User user3=new User(3,"慧芳","女",27);
    User user4=new User(4,"正伟","男",28);
    User user5=new User(5,"莉莉","女",25);
    User user6=new User(6,"敬龙","男",26);
    User user7=new User(7,"两个蝴蝶飞","男",24);
    userList.add(user1);
    userList.add(user2);
    userList.add(user3);
    userList.add(user4);
    userList.add(user5);
    userList.add(user6);
    userList.add(user7);
    return userList;
  }
}


五.二 UserServiceImpl


//指定一个 value
@Service(value="userService")
public class UserServiceImpl implements UserService{
// 用name 指定是哪一个实现, 放置在属性上
  @Resource(name="userDaoImpl")
  private UserDao userDao;
  @Override
  public List<User> listAll() {
    System.out.println("输出userDao:"+userDao);
    return userDao.listAll();
  }
}


五.三 UserAction


@Controller
public class UserAction extends ActionSupport{
  private static final long serialVersionUID = 1L;
  private UserService userService;
  /**
   * @author yuejl
   * @Description 查询全部
   * @return
   */
  public String list(){
    System.out.println("输出userService:"+userService);
    List<User> userList=userService.listAll();
    ServletActionContext.getRequest().setAttribute("userList",userList);
    return "list";
  }
  public UserService getUserService() {
    return userService;
  }
  // 放置在setter方法上。
  @Resource(name="userService")
  public void setUserService(UserService userService) {
    this.userService = userService;
  }
}


其中 @Resource 的顺序 是:

引用于: https://blog.csdn.net/wingbin/article/details/38385269


20190422195210745.png


如果指定类的话:


@Resource(name="userService",type=UserServiceImpl.class)


如果只有一个类型与其对应的话,可以直接即可。


@Resource
private UserService userService;


建议使用@Resouce


六. struts2-spring-plugin.jar


现在struts.xml 的配置依然是:


<action name="User_*" class="com.yjl.web.action.UserAction" method="{1}">
      <result name="list">list.jsp</result>
    </action>
  </package>


这个时候Dao,Service 的创建交给了Spring, 但是Action的创建并没有交给Spring, 仍然是Struts2 进行的创建。 虽然在UserAction上添加了@Controller 的注解。 这个时候,UserAction会创建两次, Spring通过注解@Controller 创建一次, Struts2通过class 全限定名称创建一次。 如果这里直接 class=“userAction” 是会出现错误的,因为 Spring 与Struts2 并没有相应的整合,两个是互不关联的。 需要将Struts2与Spring 进行整合,引入相应的jar 包,添加到classpath 路径 下即可。


20190422195828262.png


重启服务器,即可。 一定不要忘记引入整合jar 包。


<package name="user" extends="struts-default" namespace="/">
  <action name="User_*" class="userAction" method="{1}">
      <result name="list">list.jsp</result>
    </action>
  </package>


Spring 创建的 实体bean 默认的是 类名称首字母小写的样式。 相当于xml配置的:


<bean id="userAction" class="com.yjl.web.action.UserAction">


谢谢!!!

相关文章
|
3月前
|
缓存 监控 Java
SpringBoot @Scheduled 注解详解
使用`@Scheduled`注解实现方法周期性执行,支持固定间隔、延迟或Cron表达式触发,基于Spring Task,适用于日志清理、数据同步等定时任务场景。需启用`@EnableScheduling`,注意线程阻塞与分布式重复问题,推荐结合`@Async`异步处理,提升任务调度效率。
646 128
|
2月前
|
XML Java 应用服务中间件
【SpringBoot(一)】Spring的认知、容器功能讲解与自动装配原理的入门,带你熟悉Springboot中基本的注解使用
SpringBoot专栏开篇第一章,讲述认识SpringBoot、Bean容器功能的讲解、自动装配原理的入门,还有其他常用的Springboot注解!如果想要了解SpringBoot,那么就进来看看吧!
455 2
|
3月前
|
XML Java 数据格式
常用SpringBoot注解汇总与用法说明
这些注解的使用和组合是Spring Boot快速开发和微服务实现的基础,通过它们,可以有效地指导Spring容器进行类发现、自动装配、配置、代理和管理等核心功能。开发者应当根据项目实际需求,运用这些注解来优化代码结构和服务逻辑。
336 12
|
XML Java 数据格式
spring添加包扫描的方式
spring添加包扫描的方式
|
5月前
|
Java Spring 容器
SpringBoot自动配置的原理是什么?
Spring Boot自动配置核心在于@EnableAutoConfiguration注解,它通过@Import导入配置选择器,加载META-INF/spring.factories中定义的自动配置类。这些类根据@Conditional系列注解判断是否生效。但Spring Boot 3.0后已弃用spring.factories,改用新格式的.imports文件进行配置。
994 0
|
6月前
|
人工智能 Java 测试技术
Spring Boot 集成 JUnit 单元测试
本文介绍了在Spring Boot中使用JUnit 5进行单元测试的常用方法与技巧,包括添加依赖、编写测试类、使用@SpringBootTest参数、自动装配测试模块(如JSON、MVC、WebFlux、JDBC等),以及@MockBean和@SpyBean的应用。内容实用,适合Java开发者参考学习。
745 0
|
2月前
|
JavaScript Java Maven
【SpringBoot(二)】带你认识Yaml配置文件类型、SpringMVC的资源访问路径 和 静态资源配置的原理!
SpringBoot专栏第二章,从本章开始正式进入SpringBoot的WEB阶段开发,本章先带你认识yaml配置文件和资源的路径配置原理,以方便在后面的文章中打下基础
351 3
|
2月前
|
Java 测试技术 数据库连接
【SpringBoot(四)】还不懂文件上传?JUnit使用?本文带你了解SpringBoot的文件上传、异常处理、组件注入等知识!并且带你领悟JUnit单元测试的使用!
Spring专栏第四章,本文带你上手 SpringBoot 的文件上传、异常处理、组件注入等功能 并且为你演示Junit5的基础上手体验
897 2
|
9月前
|
前端开发 Java 数据库
微服务——SpringBoot使用归纳——Spring Boot集成Thymeleaf模板引擎——Thymeleaf 介绍
本课介绍Spring Boot集成Thymeleaf模板引擎。Thymeleaf是一款现代服务器端Java模板引擎,支持Web和独立环境,可实现自然模板开发,便于团队协作。与传统JSP不同,Thymeleaf模板可以直接在浏览器中打开,方便前端人员查看静态原型。通过在HTML标签中添加扩展属性(如`th:text`),Thymeleaf能够在服务运行时动态替换内容,展示数据库中的数据,同时兼容静态页面展示,为开发带来灵活性和便利性。
431 0
|
9月前
|
XML Java 数据库连接
微服务——SpringBoot使用归纳——Spring Boot集成MyBatis——基于 xml 的整合
本教程介绍了基于XML的MyBatis整合方式。首先在`application.yml`中配置XML路径,如`classpath:mapper/*.xml`,然后创建`UserMapper.xml`文件定义SQL映射,包括`resultMap`和查询语句。通过设置`namespace`关联Mapper接口,实现如`getUserByName`的方法。Controller层调用Service完成测试,访问`/getUserByName/{name}`即可返回用户信息。为简化Mapper扫描,推荐在Spring Boot启动类用`@MapperScan`注解指定包路径避免逐个添加`@Mapper`
512 0