springboot @Configuration和@Componment的區(qū)別及說明
@Configuration和@Componment的區(qū)別
@Configuration
@Target({ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Documented @Component public @interface Configuration { @AliasFor( annotation = Component.class ) String value() default ""; }
public class Car { private int id; private String name; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } } public class Driver { private int id; private String name; private Car car; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public Car getCar() { return car; } public void setCar(Car car) { this.car = car; } } import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @Configuration public class MyTestConfig { @Bean public Driver driver() { Driver driver = new Driver(); driver.setId(1); driver.setName("driver"); driver.setCar(car()); return driver; } @Bean public Car car() { Car car = new Car(); car.setId(1); car.setName("car"); return car; } } import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @Component public class MyTestConfig { @Bean public Driver driver() { Driver driver = new Driver(); driver.setId(1); driver.setName("driver"); driver.setCar(car()); return driver; } @Bean public Car car() { Car car = new Car(); car.setId(1); car.setName("car"); return car; } }
上面兩段代碼除MyTestConfig類上的注解不同之外其他都相同,但Spring對兩者的處理方式是完全不一樣的。
第一段代碼會像我們期望的一樣正常運行
因為driver()這段代碼中driver.setCar(car())方法會由Spring代理執(zhí)行,
Spring發(fā)現方法所請求的Bean已經在容器中,那么就直接返回容器中的Bean。
所以全局只有一個Car對象的實例。
第二段代碼在執(zhí)行driver()
時driver.setCar(car())不會被Spring代理,會直接調用car()方法獲取一個全新的Car對象實例,所以全局會有多個Car對象的實例
造成這種差異的原因如下:
概括就是 @Configuration 中所有帶 @Bean
注解的方法都會被動態(tài)代理,因此調用該方法返回的都是同一個實例。
其工作原理是:如果方式是首次被調用那么原始的方法體會被執(zhí)行并且結果對象會被注冊到Spring上下文中,之后所有的對該方法的調用僅僅只是從Spring上下文中取回該對象返回給調用者。
在上面的第二段代碼中
driver.setCar(car())只是純JAVA方式的調用,多次調用該方法返回的是不同的對象實例。
要修正第二段代碼中的問題,可以使用@Autowired如下所示:
import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.stereotype.Component; //@Configuration @Component public class MyTestConfig2 { @Autowired Car car; @Bean public Driver driver() { Driver driver = new Driver(); driver.setId(1); driver.setName("driver"); driver.setCar(car); return driver; } @Bean public Car car() { Car car = new Car(); car.setId(1); car.setName("car"); return car; } }
總結
1. 通過@Configuration或者@Componment注解的類默認是非懶加載,可以使用@Lazy或者在@Bean上@Lazy開啟懶加載;
2. 通過@Configuration注解的類默認會被CGLIB動態(tài)代理,所有被 @Bean 注解標記的方法將來都是通過代理方法進行調用,在該類中使用@Bean時為單列;
3. 通過@Componment注解的類默認不會產生動態(tài)代理,在該類中使用@Bean,是多實例的。
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
Spring 4.0新功能:@Conditional注解詳細介紹
Spring Boot的強大之處在于使用了Spring 4框架的新特性:@Conditional注釋,此注釋使得只有在特定條件滿足時才啟用一些配置。下面這篇文章主要給大家介紹了關于Spring4.0中新功能:@Conditional注解的相關資料,需要的朋友可以參考下。2017-09-09Mybatis如何使用動態(tài)語句實現批量刪除(delete結合foreach)
這篇文章主要介紹了Mybatis如何使用動態(tài)語句實現批量刪除(delete結合foreach),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-03-03