原因:Bean的作用域預設是單例模式的,也就是說所有⼈的使⽤的都是同⼀個物件!之前我們學單例模式的時候都知道,使⽤單例可以很⼤程度上提⾼性能,所以在 Spring 中Bean 的作⽤域預設也是 singleton 單例模式。
@Component public class Users { @Bean public User user1(){ User user = new User(); user.setId(1); user.setName("Java"); return user; } }
@Component public class Bean1 { @Autowired private User user; public User getUser(){ System.out.println("Bean1对象未修改name之前 : "+user); user.setName("C++"); return user; } }
@Component public 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()); } }
相同資源只建立一份,節省空間
不需要過多的創建和銷毀對象,執行速度提高
作用域,一般理解為:限定程式中變數的可⽤範圍叫做作⽤域,或者說在原始碼中定義變數的某個區域就叫做作⽤域。
⽽Bea
的作⽤域是指Bean
在Spring
整個框架中的某種⾏為模式,⽐如#singleton
單例作⽤域,就表
示Bean
在整個Spring
中只有⼀份,它是全域共享的,那麼當其他⼈修改了這個值之後,那麼另⼀個
⼈讀取到的就是被修改的值。
在Spring中,bean 的作用域被稱為是行為模式,因為在Spring看來,單例模式,就是一種行為,意味著在整個Spring中bean只能存在一份。
singleton:單例作⽤域
prototype:原型作⽤域(多例作⽤域)
request:請求作⽤域
session:會話作⽤域
#application:全域作⽤域
#websocket:HTTP WebSocket 作⽤域
後四種都是SpringMVC中限定使用的,因此現階段我們只學前兩個就行。
回到剛才的案例,Bean2希望取得到的bean物件是未被修改的,我們就可以將單例模式修改為多例模式。
使用@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE)
##使用@Scope ("prototype")
:當執行性到組裝Bean的屬性那一步時,當掃描到有屬性注入時,會先停下類別注入,優先進行屬性注入,因為後面的方法可能會用到該屬性。 2.1 Bean的生命週期
這個初始化方法和上面一個用註解初始化的方法是兩個不同時期的產物,init是xml時代產物,@PostConstruct是註解時代產物。優先權:當梁總方法同時存在時,優先執行註解,再執行init執行 BeanPostProcessor 初始化後置方法(如果沒有重寫此方法,請依照來源碼操作)。
@PreDestroy和destroy-method的关系和初始化方法的两个关系差不多
优先级:@ProDestroy > 重写的DisposableBean接口方法 > destroy-method
执行流程图如下:
ps:
实例化和初始化的区别:实例化
就是 分配内存空间。初始化
,就是把我们一些参数,方法的具体实现逻辑给加载进去。
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配置
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:content="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd"> <content:component-scan base-package="com.beans"></content:component-scan> <bean id="1" class="com.beans.BeanLifeComponent" init-method="init"></bean> </beans>
@Controller public class TestUser { @Autowired private Test test; public TestUser(){ test.sayHi(); System.out.println("TestUser->调用构造方法"); } }
如果这段代码先执行了初始化,也就是其构造方法,会用到test对象,此时还没有设置属性,test就为null,会造成空指针异常。因此必须先设置属性,在进行初始化。
以上是Java Spring中Bean的作用域及生命週期是什麼的詳細內容。更多資訊請關注PHP中文網其他相關文章!