Bean的作用域
Bean的作用域有很多种,在Spring Framework中支持6种(其中有四种只有在web环境中才能生效),同时Spring还支持自定义Bean的范围。
Spring Framework中支持的6种范围:
作用域 | 解释 |
singleton | 每个Spring IoC容器内同名称的bean只有一个实例(默认) |
prototype | 每次使用该bean时会创建新的实例 |
request | 每个HTTP请求生命周期内,创建新的实例(web环境中) |
session | 每个HTTP Session生命周期内,创建新的实例(web环境中) |
application | 每个ServletContext生命周期内,创建新的实例(web环境中) |
websocket | 每个WebSocket生命周期内,创建新的实例(Web环境中) |
如果想要更改 Bean 的作用域应该如何进行设置呢?
@Scope
我们可以通过修改@Scope注解中的value属性的值来修改Bean的作用域。
singleton
当我们直接在Spring中定义一个Bean时此时它默认的作用域是 singleton(单例)。
此时先定义一个类,然后将它交给Spring管理:
public class Dog {public String name;public Dog(){};public Dog(String name){this.name = name;}
}
@Component
public class Demo {@Beanpublic Dog dog() {Dog dog = new Dog();return dog;}
}
此时我们通过ApplicationContext对象来从容器中分两次拿取,看两次拿取的类是否相同。
@SpringBootApplication
public class Application {public static void main(String[] args) {//获取Spring的上下文ApplicationContext context = SpringApplication.run(Application.class, args);Object dog1 = context.getBean("dog");Object dog2 = context.getBean("dog");System.out.println(dog1);System.out.println(dog2);}
}
从打印的结果中我们可以看出dog1和dog2这两个对象是相同的。
prototype
将Bean作用域修改为 prototype 之后再进行打印观察,通过结果可以看出是两个不同的对象:
@Component
public class Demo {@Bean@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)public Dog dog() {Dog dog = new Dog();return dog;}
}
注意:request,session,application,websocket这四中作用域下的Bean只能再Web环境中获取,否则会报如下异常:
在web环境中获取Bean对象:
@RestController
@RequestMapping("/bean")
public class BeanController {@Autowiredprivate Dog dog1;@Autowiredprivate Dog dog2;@RequestMapping("/fun")public String test() {return "<p>"+dog1.toString()+"</p><p>"+dog2.toString()+"</p>";}
}
@RequestScope
被该注解修饰表示该Bean的作用域是 request 。
@Component
public class Demo {@Bean@RequestScopepublic Dog dog() {Dog dog = new Dog();return dog;}
}
当通过浏览器获取结果是发现这两个Bean是相同的,但是如果再刷新一下浏览器就会发现两次访问获取的结果不同:
@SessionScope
被该注解修饰表示该Bean的作用域为 session 。
@Component
public class Demo {@Bean@SessionScopepublic Dog dog() {Dog dog = new Dog();return dog;}
}
此时通过浏览器获取结果,可以从结果中得知同一个浏览器的结果相同,不同浏览器结果不同。
@ApplicationScope
该注解表示被修饰地Bean的作用域为 application 。
@Component
public class Demo {@Bean@ApplicationScopepublic Dog dog() {Dog dog = new Dog();return dog;}
}
Application scope就是对于整个web容器来说,bean的作用域是ServletContext级别的。这个和
singleton有点类似,区别在于:Application scope是ServletContext的单例,singleton是一个
ApplicationContext的单例,在⼀个web容器中ApplicationContext可以有多个。
Bean的生命周期
Bean的生命周期指的是Bean对象从创建直到销毁的全过程。
Bean的生命周期大致可以分为以下5步:
实例化(为Bean对象分配内存空间);
属性赋值(Bean注入或装配,如@AutoWired);
初始化;
- 执行各种通知,如 BeanNameAware, BeanFactoryAware ,ApplicationContextAware 的接口方法
- 执行初始化方法:
- xml定义 init-method
- 使用注解的方式 @PostConstruct
- 执行初始化后置方法(BeanPostProcessor )
使用Bean;
销毁Bean。
光这么看有点抽象,下面用代码展示一下:
@Slf4j
@Component
public class Animal {public Dog dog;//实例化public Animal() {log.info("执行构造方法");}//属性注入@Autowiredpublic void setDog(Dog dog) {this.dog = dog;log.info("属性注入");}//初始化@PostConstructpublic void init() {log.info("执行初始化方法");}//使用public void use() {log.info("use");}//销毁@PreDestroypublic void destroy() {log.info("执行destroy方法");}
}
接下来执行这个类中的use()方法,然后观察打印结果:
@SpringBootApplication
public class Application {public static void main(String[] args) {//获取Spring的上下文ApplicationContext context = SpringApplication.run(Application.class, args);Animal animal = (Animal) context.getBean("animal");animal.use();}
}