1.Bean的作用域
1.1 被修改的Bean案例
原因:Bean的作用域默认是单例模式的,也就是说所有⼈的使⽤的都是同⼀个对象!之前我们学单例模式的时候都知道,使⽤单例可以很⼤程度上提⾼性能,所以在 Spring 中Bean 的作⽤域默认也是 singleton 单例模式。
@Componentpublic class Users { @Bean public User user1(){ User user = new User(); user.setId(1); user.setName("Java"); return user; }}
登录后复制
@Componentpublic class Bean1 { @Autowired private User user; public User getUser(){ System.out.println("Bean1对象未修改name之前 : "+user); user.setName("C++"); return user; }}
登录后复制
@Componentpublic class Bean2 { @Autowired private User user; public User getUser(){ return user; }}
登录后复制
public class App { public static void main(String[] args) { ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml"); Bean1 bean1 = context.getBean("bean1",Bean1.class); System.out.println(bean1.getUser()); Bean2 bean2 = context.getBean("bean2",Bean2.class); System.out.println(bean2.getUser()); }}
登录后复制
1.2 为什么使用单例模式作为默认作用域
相同资源只创建一份,节省空间
不需要过多的创建和销毁对象,执行速度提高
1.3 作用域
作用域,一般理解为:限定程序中变量的可⽤范围叫做作⽤域,或者说在源代码中定义变量的某个区域就叫做作⽤域。
⽽Bea的作⽤域是指Bean在 Spring整个框架中的某种⾏为模式,⽐如singleton单例作⽤域,就表
示Bean在整个Spring中只有⼀份,它是全局共享的,那么当其他⼈修改了这个值之后,那么另⼀个
⼈读取到的就是被修改的值。
立即学习“Java免费学习笔记(深入)”;
在Spring中,bean 的作用域被称为是行为模式,因为在Spring看来,单例模式,就是一种行为,意味着在整个Spring中bean只能存在一份。
1.4 Bean的6种作用域
singleton:单例作⽤域
prototype:原型作⽤域(多例作⽤域)
request:请求作⽤域
session:会话作⽤域
application:全局作⽤域
websocket:HTTP WebSocket 作⽤域
后四种都是SpringMVC中限定使用的,因此现阶段我们只学前两个就行。
1.5 设置作用域
回到刚才的案例,Bean2希望获取到的bean对象是未被修改的,我们就可以将单例模式修改为多例模式。
使用@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
使用@Scope(“prototype”)
2.Spring执行流程和Bean的生命周期
ps:当执行性到装配Bean的属性那一步时,当扫描到有属性注入时,会先停下类注入,优先进行属性注入,因为后面的方法可能会用到该属性。
2.1 Bean的生命周期
所谓的生命周期指的是一个对象从诞生到销毁的整个生命过程,我们把这个过程就叫做一个对象的生命周期。
Bean 的生命周期分为以下5大部分:
1.实例化 Bean(为 Bean 分配内存空间)
2.设置属性(Bean 注入和装配)
3.Bean 初始化
实现了各种 Aware 通知的方法,如 BeanNameAware、BeanFactoryAware、 ApplicationContextAware 的接口方法,例如:Spring在初始化 bean,是需要给 bean 赋予 id(name)。而设置 beanName 成功的话,就会生成一个 beadNameAware 通知;执行 BeanPostProcessor 初始化前置方法(如果没有重写此方法,按照源码操作);执行 @PostConstruct 初始化方法,依赖注入操作之后被 执行;执行自己指定的 init-method 方法(如果有指定的话),是Spring中bean标签内指定的方法;
这个初始化方法和上面一个用注解初始化的方法是两个不同时期的产物,init是xml时代产物,@PostConstruct是注解时代产物。优先级:当梁总方法同时存在时,优先执行注解,再执行init执行 BeanPostProcessor 初始化后置方法(如果没有重写此方法,按照源码操作)。
4.使用 Bean
5.销毁 Bean销毁容器的各种方法, 如 @PreDestroy、DisposableBean 接口方法、destroy-method。
@PreDestroy和destroy-method的关系和初始化方法的两个关系差不多
优先级:@ProDestroy > 重写的DisposableBean接口方法 > destroy-method
执行流程图如下:
ps:实例化和初始化的区别:实例化就是 分配内存空间。初始化,就是把我们一些参数,方法的具体实现逻辑给加载进去。
2.1.1生命周期演示
xml配置如下:
Bean
public class BeanLifeComponent implements BeanNameAware { @PostConstruct public void PostConstruct(){ System.out.println("执行@PostConstruct"); } public void init(){ System.out.println("执行bean-init-method"); } public void use(){ System.out.println("正在使用bean"); } @PreDestroy public void PreDestroy(){ System.out.println("执行@PreDestroy"); } public void setBeanName(String s){ System.out.println("执行了Aware通知"); }}
登录后复制
启动类
public class App2 { public static void main(String[] args) { ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml"); BeanLifeComponent beanLifeComponent = context.getBean(BeanLifeComponent.class); beanLifeComponent.use(); context.destroy(); }}
登录后复制
xml配置
登录后复制
2.1.2 为什么要先设置属性,在进行初始化
@Controllerpublic class TestUser { @Autowired private Test test; public TestUser(){ test.sayHi(); System.out.println("TestUser->调用构造方法"); }}
登录后复制
如果这段代码先执行了初始化,也就是其构造方法,会用到test对象,此时还没有设置属性,test就为null,会造成空指针异常。因此必须先设置属性,在进行初始化。
以上就是Java Spring中Bean的作用域及生命周期是什么的详细内容,更多请关注【创想鸟】其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。
发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2627380.html