目录
2. 存储 Bean 对象 ( 添加注解存储 Bean 对象)
面试题: @Autowired 和 @Resource 的区别
在上一篇中,可以看到,创建Spring需要三步,存 bean 需要三步, 取 bean 需要三步,感觉这个读取和存储对象的步骤并没有想象的那么 "简单" 啊
所以本篇将学习通过 "使用注解" 来更简单的存储和读取对象
前面在存储 Bean 时,需要在 spring-config 中添加一行 bean 注册内容,这个XML文件有一点不好的是调试,而且XML文件报错了是不影响项目运行的,即使报错了,也可能发现不了.
- "1.0" encoding="UTF-8"?>
"http://www.springframework.org/schema/beans" - xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xmlns:content="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">
-
package="com.bit.service">
在 spring 配置文件中设置 bean 的扫描根路径
使用 5 大类注解实现,就不用在spring-config 中添加一行 bean 注册内容了
- package com.beans;
-
- import org.springframework.stereotype.Controller;
-
- @Controller
- public class UserController {
- public void sayHi() {
- System.out.println("你好,UserController!");
- }
- }
- import com.beans.UserController;
- import org.springframework.context.ApplicationContext;
- import org.springframework.context.support.ClassPathXmlApplicationContext;
-
- public class app {
- public static void main(String[] args) {
- //1. 先得到上下文对象
- ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
-
- //2. 得到 bean
- UserController controller = context.getBean("userController",UserController.class);
-
- //3. 使用 bean
- controller.sayHi();
- }
- }
- import com.beans.UserController;
- import com.beans.UserService;
- import org.springframework.context.ApplicationContext;
- import org.springframework.context.support.ClassPathXmlApplicationContext;
-
- public class app {
- public static void main(String[] args) {
- ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
- UserService service = context.getBean("userService",UserService.class);
- service.sayHi();
- }
- }
- package com.beans;
-
- import org.springframework.stereotype.Service;
-
- @Service
- public class UserService {
- public void sayHi() {
- System.out.println("你好,Service!");
- }
- }
- import com.beans.UserRepository;
- import org.springframework.context.ApplicationContext;
- import org.springframework.context.support.ClassPathXmlApplicationContext;
-
- public class app {
- public static void main(String[] args) {
- ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
- UserRepository repository = context.getBean("userRepository",UserRepository.class);
- repository.sayHi();
-
- }
- }
- package com.beans;
-
- import org.springframework.stereotype.Repository;
-
- @Repository
- public class UserRepository {
- public void sayHi() {
- System.out.println("你好,Repository!");
- }
- }
- package com.beans;
-
- import org.springframework.context.annotation.Configuration;
-
- @Configuration
- public class UserConfiguration {
- public void sayHi() {
- System.out.println("你好,Configuration!");
- }
- }
- import com.beans.UserConfiguration;
- import org.springframework.context.ApplicationContext;
- import org.springframework.context.support.ClassPathXmlApplicationContext;
-
- public class app {
- public static void main(String[] args) {
- //1. 先得到上下文对象
- ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml"); UserConfiguration configuration = context.getBean("userConfiguration",UserConfiguration.class);
- configuration.sayHi();
- }
- }
- package com.beans;
-
- import org.springframework.stereotype.Component;
-
- @Component
- public class UserComponent {
- public void sayHi() {
- System.out.println("你好,Component!");
- }
- }
- import com.beans.*;
- import org.springframework.context.ApplicationContext;
- import org.springframework.context.support.ClassPathXmlApplicationContext;
-
- public class app {
- public static void main(String[] args) {
- ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
- UserComponent component = context.getBean("userComponent",UserComponent.class);
- component.sayHi();
- }
- }
(1) 作用:
这么多注解的作用是,如果一个项目特别大的情况下,文件会非常杂乱的,所以就是通过注解,来让程序员看到注解后就知道了,当前类的作用
(2) 关系:
在五大注解类中 Component 可以当做其他注解的父类
其他类之间的关系可以当做是兄弟
下面看一下源码
在前按照规范"大驼峰"取类名,然后getBean按照规范"小驼峰"是可以运行成功的
而写如果写一个这样的类
但是如果,这样写就可以运行成功
那么 Bean 的命名规则到底应该是什么样的
所以根据这个规则,正常下命名,如果类的前两个字符都为大写字母,就写原名字
如果只有第一个字母为大写,那就按"小驼峰" 来命名
@Bean 是加在方法上的,并且只使用一个 @Bean 是无法将对象存储到容器中的,所以还要给类加个注解
Bean 就是类注解的基础上缩小范围
- package com.beans;
-
- import org.springframework.context.annotation.Bean;
- import org.springframework.stereotype.Component;
-
- @Component
- public class UserBeans {
- @Bean //[只使用一个 @Bean 是无法将对象存储到容器中]
- public User user1() {
- User user = new User();
- user.setId(1);
- user.setName("张三");
- return user;
- }
- }
- import com.beans.*;
- import org.springframework.context.ApplicationContext;
- import org.springframework.context.support.ClassPathXmlApplicationContext;
-
- public class app {
- public static void main(String[] args) {
- //1. 先得到上下文对象
- ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
- User user = context.getBean("user1",User.class);
- // User user = context.getBean(User.class);
-
- System.out.println(user);
-
- }
- }
- package com.beans;
-
- public class User {
- private int id;
- private String name;
-
- @Override
- public String toString() {
- return "User{" +
- "id=" + id +
- ", name='" + name + '\'' +
- '}';
- }
-
- public int getId() {
- return id;
- }
-
- public void setId(int id) {
- this.id = id;
- }
-
- public String getName() {
- return name;
- }
-
- public void setName(String name) {
- this.name = name;
- }
- }
可以通过设置 name 属性给 Bean 对象进⾏重命名操作
- package com.beans;
-
- import org.springframework.context.annotation.Bean;
- import org.springframework.stereotype.Component;
-
- @Component
- public class UserBeans {
- @Bean(name = "userinfo") //[只使用一个 @Bean 是无法将对象存储到容器中]
- public User user1() {
- User user = new User();
- user.setId(1);
- user.setName("张三");
- return user;
- }
- }
@Bean 命名规则,当没有设置name属性时,那么 bean 默认的名称就是方法名,
当设置name属性后,只能通过重命名的name属性对应的值来获取,也就是说再使用方法就获取不到 bean 对象了,
并且也可以 一个bean 有多个名字,像数组一样在{}中写上多个名字,就可以了
获取 Bean 对象也叫做对象装配,是把对象取出来放到某个类中,有时候也叫对象注入
通过 @Autowired 就可以属性注入
- package com.beans;
-
- import org.springframework.beans.factory.annotation.Autowired;
- import org.springframework.stereotype.Controller;
-
- @Controller
- public class UserController2 {
- // 对象注入1: 属性注入
- @Autowired
- private UserService userService;
-
- public void sayHi() {
- userService.sayHi();
- }
- }
- import com.beans.*;
- import org.springframework.context.ApplicationContext;
- import org.springframework.context.support.ClassPathXmlApplicationContext;
-
- public class app {
- public static void main(String[] args) {
- //1. 先得到上下文对象
- ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
- UserController2 controller2 = context.getBean(UserController2.class);
- controller2.sayHi();
- }
- }
写构造方法,参数传入要注入的对象,然后写上注解 @AutoWired (如果当前类中只有一个构造方法,那就可以省略注解)
- package com.beans;
-
- import org.springframework.beans.factory.annotation.Autowired;
- import org.springframework.stereotype.Controller;
-
- @Controller
- public class UserController2 {
-
- //使用构造方法实现 bean 注入(官方推荐写法)
- private UserService userService;
-
- @Autowired
- public UserController2(UserService userService) {
- // userService = new UserService(); //传统的写法
- this.userService = userService;
- }
-
- public void saHi() {
- userService.sayHi();
- }
- }
Setter 注入,写要注入的属性的 setter方法,然后加上注解 @Autowired
- package com.beans;
-
- import org.springframework.beans.factory.annotation.Autowired;
- import org.springframework.stereotype.Controller;
-
- @Controller
- public class UserController2 {
- //3. 使用 Setter 注入
- private UserService userService;
-
- @Autowired
- public void setUserService(UserService userService) {
- this.userService = userService;
- }
-
- public void sayHi() {
- userService.sayHi();
- }
- }
属性注入 / 构造方法注入 / Setter注入
- 属性注入特点: 写法简单,只适用于 IoC容器,通用性不好
- 构造方法注入: 这种写法从 spring 3.4后就成推荐写法了,通用性更好,能确保在使用注入对象之前,此注入对象一定是初始化过的.当构造方法注入参数过多时,开发者就要检查自己所写的代码是否符合单一设计原则的规范了.
- Setter 注入特点:这种写法是早期 Spring 版本的推荐写法(大概是3.4前),Setter 注入通用性没有构造方法注入通用
@Resource: 支持 属性注入 和 Setter 注入,但不支持构造方法注入
相同点: 都可以实现将一个对象注入到类中不同点:
- 出身不同: @Resource 来自 JDK ; @Autowired 是 Spring 框架提供的
- 用法不同: @Resource 用于 属性注入/Setter 注入 ; @ Autowired 用于 属性注入/构造方法注入/Setter注入
- 支持的参数不同: @Resource 支持更多的参数设置,name/type.... ; 而 @Autowired 只支持 required 参数设置
解决方法
(1) 精确的描述 bean 的名称 (将注入的名称写对)
(2) 使用 @Resource 设置 name 的方式来重命名注入对象
(3) 如果限定不能删除 @Autowired ,那就可以再加上使用 @Qualifier,来删选 bean 对象
解决方法
(1) 精确的描述 bean 的名称 (将注入的名称写对)
- package com.beans;
-
- import org.springframework.beans.factory.annotation.Autowired;
- import org.springframework.beans.factory.annotation.Qualifier;
- import org.springframework.stereotype.Controller;
-
- @Controller
- public class UserController {
-
- @Autowired //[属性注入/字段注入]
- private User user2;
-
- public void sayHi() {
- System.out.println("User -> " + user2);
- }
- }
- import com.beans.*;
- import org.springframework.context.ApplicationContext;
- import org.springframework.context.support.ClassPathXmlApplicationContext;
-
- public class app {
- public static void main(String[] args) {
- //1. 先得到上下文对象
- ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
- UserController controller = context.getBean(UserController.class);
- controller.sayHi();
- }
- }
- package com.beans;
-
- import org.springframework.context.annotation.Bean;
- import org.springframework.stereotype.Component;
-
- @Component
- public class UserBeans {
- @Bean(name = "userinfo") //[只使用一个 @Bean 是无法将对象存储到容器中]
- public User user1() {
- User user = new User();
- user.setId(1);
- user.setName("张三");
- return user;
- }
-
- @Bean
- public User user2() {
- User user = new User();
- user.setId(2);
- user.setName("李四");
- return user;
- }
- }
(2) 使用 @Resource 设置 name 的方式来重命名注入对象
(3) 如果限定不能删除 @Autowired ,那就可以再加上使用 @Qualifier,来删选 bean 对象