您好,登錄后才能下訂單哦!
本文小編為大家詳細介紹“Java Spring中Bean的作用域及生命周期是什么”,內容詳細,步驟清晰,細節(jié)處理妥當,希望這篇“Java Spring中Bean的作用域及生命周期是什么”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來學習新知識吧。
原因: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()); } }
相同資源只創(chuàng)建一份,節(jié)省空間
不需要過多的創(chuàng)建和銷毀對象,執(zhí)行速度提高
作用域,一般理解為:限定程序中變量的可?范圍叫做作?域,或者說在源代碼中定義變量的某個區(qū)域就叫做作?域。
?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")
ps
:當執(zhí)行性到裝配Bean的屬性那一步時,當掃描到有屬性注入時,會先停下類注入,優(yōu)先進行屬性注入,因為后面的方法可能會用到該屬性。
所謂的生命周期指的是一個對象從誕生到銷毀的整個生命過程,我們把這個過程就叫做一個對象的生命周期。
Bean 的生命周期分為以下5大部分:
1.實例化 Bean(為 Bean 分配內存空間)
2.設置屬性(Bean 注入和裝配)
3.Bean 初始化
實現了各種 Aware 通知的方法,如 BeanNameAware、BeanFactoryAware、 ApplicationContextAware 的接口方法,例如:Spring在初始化 bean,是需要給 bean 賦予 id(name)。而設置 beanName 成功的話,就會生成一個 beadNameAware 通知;執(zhí)行 BeanPostProcessor 初始化前置方法(如果沒有重寫此方法,按照源碼操作);執(zhí)行 @PostConstruct 初始化方法,依賴注入操作之后被 執(zhí)行;執(zhí)行自己指定的 init-method 方法(如果有指定的話),是Spring中bean標簽內指定的方法;
這個初始化方法和上面一個用注解初始化的方法是兩個不同時期的產物,init是xml時代產物,@PostConstruct是注解時代產物。優(yōu)先級:當梁總方法同時存在時,優(yōu)先執(zhí)行注解,再執(zhí)行init執(zhí)行 BeanPostProcessor 初始化后置方法(如果沒有重寫此方法,按照源碼操作)。
4.使用 Bean
5.銷毀 Bean銷毀容器的各種方法, 如 @PreDestroy、DisposableBean 接口方法、destroy-method。
@PreDestroy和destroy-method的關系和初始化方法的兩個關系差不多
優(yōu)先級:@ProDestroy > 重寫的DisposableBean接口方法 > destroy-method
執(zhí)行流程圖如下:
ps:
實例化和初始化的區(qū)別:實例化
就是 分配內存空間。初始化
,就是把我們一些參數,方法的具體實現邏輯給加載進去。
xml配置如下:
Bean
public class BeanLifeComponent implements BeanNameAware { @PostConstruct public void PostConstruct(){ System.out.println("執(zhí)行@PostConstruct"); } public void init(){ System.out.println("執(zhí)行bean-init-method"); } public void use(){ System.out.println("正在使用bean"); } @PreDestroy public void PreDestroy(){ System.out.println("執(zhí)行@PreDestroy"); } public void setBeanName(String s){ System.out.println("執(zhí)行了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->調用構造方法"); } }
如果這段代碼先執(zhí)行了初始化,也就是其構造方法,會用到test對象,此時還沒有設置屬性,test就為null,會造成空指針異常。因此必須先設置屬性,在進行初始化。
讀到這里,這篇“Java Spring中Bean的作用域及生命周期是什么”文章已經介紹完畢,想要掌握這篇文章的知識點還需要大家自己動手實踐使用過才能領會,如果想了解更多相關內容的文章,歡迎關注億速云行業(yè)資訊頻道。
免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。