目录
一、概述
对于 DI 使用注解,将不再需要在 Spring 配置文件中声明 bean 实例。
Spring 中使用注解,需要在原有 Spring 运行环境基础上再做一些改变。
需要在 Spring 配置文件中配置组件扫描器,用于在指定的基本包中扫描注解。
<?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 https://www.springframework.org/schema/context/spring-context.xsd">
<!--声明组件扫描器:指定注解所在包名 -->
<context:component-scan base-package="com.fancy"></context:component-scan>
</beans>
其中 spring-context.xsd 约束文件是创建组件扫描器时自动导入的
指定多个包的三种方式:
(1) 使用多个 context:component-scan 指定不同的包路径
<context:component-scan base-package="com.fancy"></context:component-scan>
<context:component-scan base-package="java.io"></context:component-scan>
(2) 指定 base-package 的值使用分隔符
分隔符可以使用逗号(,)分号(;)还可以使用空格,不建议使用空格。
<context:component-scan base-package="com.fancy,java.io"></context:component-scan>
(3) base-package 是指定到父包名
base-package 的值表是基本包,容器启动会扫描包及其子包中的注解,当然也会扫描到子包下级的子包。所以 base-package 可以指定一个父包就可以。
<context:component-scan base-package="com"></context:component-scan>
但不建议使用顶级的父包,扫描的路径比较多,导致容器启动时间变慢。指定到目标包和合适的。也就是注解所在包全路径。
二、定义 Bean 的注解 @Component
需要在类上使用注解 @Component,该注解的 value 属性用于指定该 bean 的 id 值。
另外,Spring 还提供了 3 个创建对象的注解
➢ @Repository 用于对 DAO 实现类进行注解
➢ @Service 用于对 Service 实现类进行注解
➢ @Controller 用于对 Controller 实现类进行注解
这三个注解与@Component 都可以创建对象,但这三个注解还有其他的含义,@Service创建业务层对象,业务层对象可以加入事务功能,@Controller 注解创建的对象可以作为处理器接收用户的请求。@Repository,@Service,@Controller 是对@Component 注解的细化,标注不同层的对象。即持久层对象,业务层对象,控制层对象。
@Component 不指定 value 属性,bean 的 id 是类名的首字母小写
比如这个就是创建了 id 为 student 的 bean 对象
三、简单类型属性注入 @Value
需要在属性上使用注解@Value,该注解的 value 属性用于指定要注入的值。使用该注解完成属性注入时,类中无需
setter。当然,若属性有 setter,则也可将其加到 setter 上
例如
@Value("Zhan")
private String name;
@Value("18")
private int age;
四、byType 自动注入@Autowired
需要在引用属性上使用注解@Autowired,该注解默认使用 按类型自动装配 Bean 的方式。使用该注解完成属性
注入时,类中无需 setter。当然,若属性有 setter,则也可将其加到 setter 上。
五、byName 自动注入 @Autowired 与 @Qualifier
需要在引用属性上联合使用注解 @Autowired 与 @Qualifier。@Qualifier 的 value 属性用于指定要匹配的 Bean 的
d 值。类中无需 set 方法,也可加到 set 方法上。
六、JDK 注解 @Resource 自动注入
Spring提供了对 jdk中@Resource注解的支持。@Resource 注解既可以按名称匹配Bean,也可以按类型匹配
Bean。默认是按名称注入。使用该注解,要求 JDK 必须是 6 及以上版本。@Resource 可在属性上,也可在 set 方
法上。
但是JDK11以后完全移除了javax扩展,所以此版本及其之后的jdk不能使用 @Resource 注解
如果一定想用的话,可通过maven引入依赖
<dependency>
<groupId>javax.annotation</groupId>
<artifactId>javax.annotation-api</artifactId>
<version>1.3.2</version>
</dependency>
这样就可以用了
(1) byType 注入引用类型属性
@Resource 注解若不带任何参数,采用默认按名称的方式注入,按名称不能注入 bean, 则会按照类型进行 Bean 的匹配注入。
@Component("mySchool")
public class School {
.....
}
@Resource
private School school;
在这里首先按名称无法注入bean,因为School对象名称为mySchool而不是school,然后在按照类型进行bean注入
(2) byName 注入引用类型属性
@Resource 注解指定其 name 属性,则 name 的值即为按照名称进行匹配的 Bean 的 id。
@Resource(name="mySchool")
private School school;
七、注解与 XML 的对比
注解优点是:
- 方便
- 直观
- 高效(代码少,没有配置文件的书写那么复杂)。
其弊端也显而易见:
- 以硬编码的方式写入到 Java 代码中,修改是需要重新编译代码的。
XML 方式优点是:
- 配置和代码是分离的
- 在 xml 中做修改,无需编译代码,只需重启服务器即可将新的配置加载。
xml 的缺点是:
- 编写麻烦,效率低,大型项目过于复杂