上一节我们介绍了五大类注解,这一节介绍方法注解@Bean,@Bean作用的对象是方法,该注解需要搭配五大类注解同时进行使用,因为类方法的数量远远大于类的数量,如果使用@Bean注解标记方法的类没有被标记,那么Spring Boot项目在启动时需要遍历所有的类的所有方法,开销无疑是巨大的,但如果只遍历用五大类注解标记的类的方法,无疑大大减小了遍历范围
@Bean 类注解默认情况下,Bean name = 方法名,但是方法名是非常容易重复的,很可能在两个类中有两个相同的方法,它们通过方法类注解返回同一个类的对象,若这两个对象内部属性不相同,则可能出现误调的情况(本想调用A类的student方法返回名为张三的学生的,结果调用成B类的student方法返回了一个名为李四的学生)
并且这种错误并不会引起报错,一旦出现错误非常难以排查问题,而针对上述问题可以通过给@Bean设置name属性获取对象
@Controller
public class StudentBeans {
@Bean(name = {"s1", "s2"})
public Student student() {
// 伪代码构建对象
Student stu = new Student();
stu.setId(1);
stu.setName("张三");
stu.setAge(18);
return stu;
}
}
注:当给一个@Bean设置了name属性后,就无法使用方法名获取Bean对象了,只能通过设置的名称获取
获取Bean对象也叫做对象装配,就是把注入到Spring Boot中的某个对象取出来取出来放到指定类中,这种方法也叫做对象注入,对象装配的方法有三种,接下来将详细介绍
属性注入是使用@Autowired 来实现的,将Service类注入到Controller类中
属性注入实例: 将StudentService对象注入到StudentController类中
package com.demo.controller;
import com.demo.service.StudentService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
@Controller
public class StudentController {
@Autowired
private StudentService studentService;
public void sayHi() {
studentService.sayHi();
}
}
package com.demo.service;
import org.springframework.stereotype.Controller;
import org.springframework.stereotype.Service;
@Service
public class StudentService {
public void sayHi() {
System.out.println("StudentService sayHi");
}
}
接下来来介绍属性注入的缺点:
1、功能性问题:无法注入final对象。首先这个应该是jdk的问题而并非spring的问题,final修饰的变量不可被改变,一旦获取了初始值就不能重新被赋值,如果在类中使用final属性的成员,要么直接赋值,要么在构造函数中赋值
private StudentService studentService; // 添加final关键字,无法赋值
private final int number; // 报错:必须要直接进行赋值,或者写一个构造函数,在构造函数中赋值
2、通用性问题:只能适用于IoC容器
3、设计原则问题:更容易违背单一设计原则
完全符合单一设计原则,每一个Setter方法只针对一个对象,但是它的缺点也很明显,不能注入不可变对象,注入的对象可以被任意修改
使用方法:
// 2、Setter注入
private StudentService studentService;
@Autowired
public void setStudentService(StudentService studentService) {
this.studentService = studentService;
}
Spring Boot 官方推荐的用法,如果当前类中只有一个构造方法,那么@Autowired注解可以省略
基本使用方法
// 3、构造方法注入
private StudentService studentService;
public StudentController(StudentService studentService) {
this.studentService = studentService;
}
构造方法注入的优点
1.可注入不可变对象,不可变对象可以在构造函数中初始化
2.注入对象不会被修改:构造方法在对象创建时只会执行一次,因此它不存在注入对象被随时(调用)修改的情况
3.注入对象会被完全初始化
4.通用性更好:构造方法注入可适用于任何环境,无论是 IoC 框架还是非 IoC 框架,构造方法注入的代码都是通用的,所以它的通用性更好。
功能:两者都是用来实现依赖注入的,功能非常相近
@Resource 和 @Autowired 的区别
由于@Autowired支持的参数设置很少,所以产生了@Qualifier
注解来扩充@Autowired 组件的功能
场景:使用@Bean注解向Spring中注入两个Student对象,当我们想要从Spring中取出对象,注入到其他对象中时,Spring就会不确定到底使用哪一个对象
package com.demo.componect;
import com.demo.model.Student;
import org.springframework.context.annotation.Bean;
import org.springframework.stereotype.Controller;
@Controller
public class StudentBeans {
@Bean(name = {"s1", "s2"})
public Student student1() {
// 伪代码构建对象
Student stu = new Student();
stu.setId(1);
stu.setName("张三");
stu.setAge(18);
return stu;
}
@Bean
public Student student2() {
Student stu = new Student();
stu.setId(2);
stu.setName("李四");
stu.setAge(20);
return stu;
}
}
@Resource 解决方式:
可以根据@Bean注解时设置的名称,来确定应该调用哪个@Bean注解标记的方法来获取指定对象
@Resource(name = "s1")
private Student student;
@Autowired 解决方式:
可以根据 注入方法名 = 对象名称的方式来获取对象(不推荐)
@Autowired
private Student student2;
使用@Qualifier注解可以对获取对象的方法进行筛选
@Autowired
@Qualifier("student2")
private Student student;
作用域是Bean对象在整个Spring框架(项目)中的某种行为模式,Bean对象默认情况下是单例状态(singleton),也就是所有人使用的都是同一个Bean对象,所以Spring中的Bean的作用域默认也是单例模式(singleton)
示例:
@Controller
public class UserController {
@Autowired
private User user1;
public void getUser() {
System.out.println("User1:" + user1);
User u = user1;
u.setName("李四");
System.out.println("u:" + u);
}
}
@Controller
public class UserAdviceController {
@Resource
private User user1;
public void getUser() {
System.out.println("王五 | user1: " + user1);
}
UserController uc = context.getBean("userController", UserController.class);
uc.getUser();
UserAdviceController ua = context.getBean("userAdviceController", UserAdviceController.class);
ua.getUser();
输出:
User1:User(id=1, name=张三, password=15157722660)
u:User(id=1, name=李四, password=15157722660)
王武 | user1: User(id=1, name=李四, password=15157722660)
可以看到不管是通过创建局部变量,还是重新装配,获取到的都是同一个Bean对象,这说明Bean对象全局只有一份,这个对象是一个单例
Spring容器在初始化Bean实例时,同时会指定该实例的作用域,Spring有6种作用域,最后四种是基于Spring MVC生效的:
单例作用域(singleton) vs 全局作用域(application)
@Scope标签可以用来声明Bean的作用域,比如设置Bean的作用域,如下代码所示:
@Data
@Controller
public class UserBeans {
// 方法1:
// @Scope("prototype")
// 方法2:
@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
@Bean
public User user1() {
User user = new User();
user.setName("张三");
user.setId(1);
user.setPassword("15157722660");
return user;
}
}
所谓生命周期指的是一个对象从诞生到销毁的整个生命过程,我们将这一过程叫做一个Bean对象的生命周期
实例化Bean,将字节码转换成内存中的对象(即加载,为Bean分配内存空间)
设置属性(Bena注入和装配)
Bean初始化:实现了各种Aware通知方法,如@BeanPostProcessor(注解初始化前置方法),@PostConstruct xml初始化方法(依赖注入后被执行)
使用Bean对象
销毁Bean对象