前言:
大家好,我是良辰丫,我们已经学会了Spring的存取,今天我们将一起来学习Bean对象的作用域和生命周期.
个人主页:良辰针不戳
所属专栏:javaEE进阶篇之框架学习
励志语句:生活也许会让我们遍体鳞伤,但最终这些伤口会成为我们一辈子的财富。
期待大家三连,关注,点赞,收藏。
作者能力有限,可能也会出错,欢迎大家指正。
愿与君为伴,共探Java汪洋大海。
- 所谓
Bean的作用域
是Bean在spring框架中(或者项目中)的某种行为,这是专业术语.- 其实我们从接触编程开始,我们就已经接触了作用域,全局变量和局部变量可以理解为不同的作用域.
接下来,我们来看一个简单的例子,来看一下Bean对象作用域可能出现的问题,在此之前,我们先了解idea里面的一个插件Lombok
.
进入maven中央仓库搜索Lombok,点击使用量最多的,复制依赖,引入到idea的pom.xml中.
<!-- https://mvnrepository.com/artifact/org.projectlombok/lombok -->
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.18.24</version>
<scope>provided</scope>
</dependency>
Lombok里面有很多注解,后面会详细介绍,咱们先来简单介绍一个set和get的注解.
import lombok.Data;
@Data
public class Student {
private int id;
private String name;
}
因为作用域会出现问题,我们才会进行研究它,那么我们先来看一个简单的例子吧.
代码如下,大家也可以去运行一下,观察结果.
package com.demo.Controller;
import lombok.Data;
@Data
public class User {
private int id;
private String name;
private String password;
}
package com.demo.Controller;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;
/**
* 作者:张三
*/
@Component
public class UserBeans {
@Bean
public User user() {
User user = new User();
user.setId(1);
user.setName("张三");
user.setPassword("12345");
return user;
}
}
package com.demo.Controller;
import com.demo.Controller.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
/**
* author:李四
*/
@Controller
public class UserController {
@Autowired
private User user;
public void getUser() {
System.out.println("张三 : " + user);
User u = user;
u.setName("李四");
System.out.println("李四 : " + u);
}
}
package com.demo.Controller;
import org.springframework.stereotype.Controller;
import javax.annotation.Resource;
/**
* author:王五
*/
@Controller
public class UserController2 {
@Resource
private User user;
public void getUser() {
System.out.println("王五 : " + user);
}
}
import com.demo.Controller.UserController2;
import com.demo.Controller.UserController;
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.xml");
UserController user1 = context.getBean("userController", UserController.class);
user1.getUser();
UserController2 user2 = context.getBean("userController2", UserController2.class);
user2.getUser();
}
}
运行结果如下 :
简单分析 :
- 张三给对象赋初值,通过张三这个类,打印出来的还是张三.
- 把张三的类注入到李四里面,李四通过一个变量赋值张三的所有信息,然后修改名字为李四,打印的是李四,没有问题.
- 王五把张三这里类注入到自己这里,没做任何操作,但是结果是张三的信息.
总结 :
- 之所以会发生上面的情况,我们的Bean对象,默认情况下是单例模式,所有的人都会使用相同的对象,因此张三,李四,王五其实共有一个对象.
- 单例模式可以提高性能,因此spring的Bean对象作用域默认为单例模式,只是猜测官方这样设计哈.
我们经常使用的作用域有6种,我在使用Bean对象的时候,通常会指定作用域,前两种是我们这篇文章要讲的,后面四种是在Spring MVC中涉及.
- 官⽅说明:(Default) Scopes a single bean definition to a single object instance for each Spring IoC container.
- 描述:该作⽤域下的Bean在IoC容器中只存在⼀个实例:获取Bean(即通过applicationContext.getBean等⽅法获取)及装配Bean(即通过@Autowired注⼊)都是同⼀个对象。
- 场景:通常⽆状态的Bean使⽤该作⽤域。⽆状态表示Bean对象的属性状态不需要更新.
- 备注:Spring默认选择该作⽤域
- 官⽅说明:Scopes a single bean definition to any number of object instances.
- 描述:每次对该作⽤域下的Bean的请求都会创建新的实例:获取Bean(即通过applicationContext.getBean等⽅法获取)及装配Bean(即通过@Autowired注⼊)都是新的对象实例。
- 场景:通常有状态的Bean使用该作用域.
- 官⽅说明:Scopes a single bean definition to the lifecycle of a single HTTP request. That is,each HTTP request has its own instance of a bean created off the back of a single bean
definition. Only valid in the context of a web-aware Spring ApplicationContext.- 描述:每次http请求会创建新的Bean实例,类似于prototype
- 场景:⼀次http的请求和响应的共享Bean
- 备注:限定SpringMVC中使⽤
- 官⽅说明:Scopes a single bean definition to the lifecycle of an HTTP Session. Only valid in the context of a web-aware Spring ApplicationContext.
- 描述:在⼀个http session中,定义⼀个Bean实例
- 场景:⽤户回话的共享Bean, ⽐如:记录⼀个⽤户的登陆信息
- 备注:限定SpringMVC中使⽤
- 官⽅说明:Scopes a single bean definition to the lifecycle of a ServletContext. Only valid in the context of a web-aware Spring ApplicationContext.
- 描述:在⼀个http servlet Context中,定义⼀个Bean实例
- 场景:Web应⽤的上下⽂信息,⽐如:记录⼀个应⽤的共享信息
- 备注:限定SpringMVC中使⽤
- 官⽅说明:Scopes a single bean definition to the lifecycle of a WebSocket. Only valid in the context of a web-aware Spring ApplicationContext.
- 描述:在⼀个HTTP WebSocket的⽣命周期中,定义⼀个Bean实例
- 场景:WebSocket的每次会话中,保存了⼀个Map结构的头信息,将⽤来包裹客户端消息头。第⼀次初始化后,直到WebSocket结束都是同⼀个Bean。
- 备注:限定Spring WebSocket中使⽤
使⽤ @Scope 标签就可以⽤来声明 Bean 的作⽤域.
那么,我们来看一下运行结果.
王五调用张三的实例,虽然李四在此之前已经把名字修改为李四了,但是王五调用的时候,又创建了新的实例,结果仍然是张三.
设置作用域的两种方式:
- 直接设置值:@Scope(“prototype”)
- 使⽤枚举设置(全局变量的方式):@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
小结Bean 执⾏流程(Spring 执⾏流程):
- 启动 Spring 容器
- 实例化 Bean(分配内存空间,从⽆到有)
- Bean 注册到 Spring 中(存操作)
- 将 Bean 装配到需要的类中(取操作)。
⽣命周期
指的是⼀个对象从诞⽣到销毁的整个⽣命过程,我们把这个过程就叫做⼀个对象的⽣命周期.
- 实例化Bean对象(对应JVM中的加载,从无到有,将字节码转换成内存中的对象,只是分配了内存).比喻成买了一个空壳房子.
- 设置属性(Bean对象的注入和装配).购买装修材料.
- Bean的初始化.房子进行装修.
实现了各种Aware通知的方法.
初始化的前置工作.
执行@PostConstruct初始化方法,依赖注入操作之后被执行.
执行自己指定的init-method方法
执行BeanPostProcessor初始化的后置方法.- 使用Bean.
- 销毁Bean.
实例化和初始化有什么区别呢?
- 实例化和属性设置是 Java 级别的系统“事件”,其操作过程不可⼈⼯⼲预和修改;
- ⽽初始化是给开发者提供的,可以在实例化之后,类加载完成之前进⾏⾃定义“事件”处理。
为什么先设置属性再进行初始化呢?
就像房子,我们有了装修材料才能进行装修,这也是一样,类比一下,有了属性才能进行初始化,什么属性都没有,给谁进行初始化呢?
下面的例子是为了证明先注入依赖,然后再进行初始化.
package com.demo.service;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import javax.annotation.PostConstruct;
@Service
public class AService {
@Autowired
private BService bComponent;
@PostConstruct
public void postConstruct() {
//bComponent.sayHi(); // 先 ① 再 ② 不会有问题,但如果先 ② 再 ① 就会空指针报错
System.out.println("执行了 A 对象的 PostConstruct 方法");
}
}
package com.demo.service;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import javax.annotation.PostConstruct;
@Service
public class BService {
@Autowired
private CService component;
@PostConstruct
public void postConstruct() {
System.out.println("执行了 B 对象的 PostConstruct 方法");
}
public void sayHi() {
System.out.println("hi,BComponent~");
}
}
package com.demo.service;
import org.springframework.stereotype.Service;
import javax.annotation.PostConstruct;
@Service
public class CService {
@PostConstruct
public void postConstruct() {
System.out.println("执行了 C 对象的 PostConstruct 方法");
}
}
public static void main(String[] args) {
ApplicationContext context =
new ClassPathXmlApplicationContext("spring.xml");
AService aService = context.getBean("AService",AService.class);
//aService.postConstruct();
}
- 我们调用A,但是A注入了B的依赖,那么进入 B,B又注入了C的依赖.
- 然后反过来一次执行CBA,就像套娃的过程.