目录
一、什么是Bean的作用域
二、@Scope注解
三、Bean的6种作用域
3.1 singleton单例模式
3.2 prototype 原型模式
3.3 request
3.4 session
3.5 application
3.6 websocket
在之前学习的过程中,我们把作用域定义为:限定程序中变量的可用范围,或者说是定义变量的某个区域。但是,在Bean中,这个作用域指的是Bean在Spring框架中的某种行为。接下来以一个案例分析一下
首先创建一个User类,同时创建一个Users类将User对象储存到Spring中,这里都是利用注解的方式,以后也是采取这种方式了
public class User {
private String name;
private int id;
@Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", id=" + id +
'}';
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
}
@Component
public class Users {
@Bean
public User user() {
User user = new User();
user.setName("公共名");
user.setId(1);
return user;
}
}
这里有一个需求就是一个开发人员需要获取到这个User类的对象,并将这个User类对象中原始的“公共名”改成张三,并且不能修改原类名!
在这个操作中,开发人员并没有直接拿user对象去修改其中的属性,而是另外创建了一个user1去接收user同时在user1中对name属性进行修改
@Controller
public class UserController1 {
@Resource
private User user;
public void setName() {
User user1 = user;
user1.setName("张三");
System.out.println(user1);
}
}
通过在Spring的配置文件中配置注解扫描路径,然后再通过Spring提供的工厂获取到类对象。通过测试我们发现已经是成功了。
@Test
public void test() {
ApplicationContext ctx = new ClassPathXmlApplicationContext("spring-config.xml");
UserController1 userController1 = ctx.getBean("userController1", UserController1.class);
userController1.setName();
}
按照思路这里应该是没有修改原来的user类的,但是突然有一天另一位开发人员需要去获取原来user类中的内容
@Controller
public class UserController2 {
@Resource
private User user;
public void getUser() {
System.out.println(user);
}
}
public void test() {
ApplicationContext ctx = new ClassPathXmlApplicationContext("spring-config.xml");
// 这里是第一个开发人员的
UserController1 userController1 = ctx.getBean("userController1", UserController1.class);
userController1.setName();
// 第二个开发人员的
UserController2 userController2 = ctx.getBean("userController2", UserController2.class);
userController2.getUser();
}
这里测试就会发现有问题了,原始类中的name也被修改了,其实这就是我们说的Bean的作用域,因为Bean作用域默认是Singleton,所以这里只会创建出一个对象!
对于上述的问题我们要怎么去解决呢?我们这里引入一个新的注解@Scope来设置Bean的作用域,这里将Bean的作用域设置为“prototype”原型模式(多例模式)
@Component
public class Users {
@Bean("user")
@Scope("prototype")
public User user() {
User user = new User();
user.setName("公共名");
user.setId(1);
return user;
}
}
再次运行测试程序,我们发现这个问题就解决了
除了直接写上作用域的方式@Scope还提供了一种添加作用域的方式:使用枚举设置,这种方式和直接写作用域名称的方式是等价的
@Component
public class Users {
@Bean("user")
@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
public User user() {
User user = new User();
user.setName("公共名");
user.setId(1);
return user;
}
}
描述:该作用域下的Bean在IoC容器中只有一个实例:获取Bean(即通过
applicationContext.getBean等⽅法获取)及装配Bean(即通过@Autowired注⼊)都是同⼀个对象场景:通常⽆状态的Bean使⽤该作⽤域。⽆状态表示Bean对象的属性状态不需要更新
备注:Spring默认选择该作用域
描述:每次对该作⽤域下的Bean的请求都会创建新的实例:获取Bean(即通过applicationContext.getBean等⽅法获取)及装配Bean(即通过@Autowired注⼊)都是新的对象实例。
场景:通常有状态的Bean使⽤该作⽤域
描述:每次http请求会创建新的Bean实例,类似于prototype
场景:⼀次http的请求和响应的共享Bean
备注:限定SpringMVC中使⽤
描述:在⼀个http session中,定义⼀个Bean实例
场景:⽤户回话的共享Bean, ⽐如:记录⼀个⽤户的登陆信息
备注:限定SpringMVC中使⽤
描述:在⼀个http servlet Context中,定义⼀个Bean实例
场景:Web应⽤的上下⽂信息,⽐如:记录⼀个应⽤的共享信息
备注:限定SpringMVC中使⽤
描述:在⼀个HTTP WebSocket的⽣命周期中,定义⼀个Bean实例场景:WebSocket的每次会话中,保存了⼀个Map结构的头信息,将⽤来包裹客户端消息头。第⼀次初始化后,直到WebSocket结束都是同⼀个Bean。
备注:限定Spring WebSocket中使⽤