您好,登錄后才能下訂單哦!
這篇文章主要講解了“java中@Configuration使用場景是什么”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“java中@Configuration使用場景是什么”吧!
@Configuration注解可以標注到類上,當標注到類上時,啟動Spring就會自動掃描@Configuration注解標注的類,將其注冊到IOC容器中,并被實例化成Bean對象。
如果被@Configuration注解標注的類中存在使用@Bean注解標注的創建某個類對象的方法,那么,Spring也會自動執行使用@Bean注解標注的方法,將對應的Bean定義信息注冊到IOC容器,并進行實例化。
@Configuration注解是從Spring 3.0版本開始加入的一個使Spring能夠支持注解驅動開發的標注型注解,主要用于標注在類上。當某個類標注了@Configuration注解時,表示這個類是Spring的一個配置類。@Configuration注解能夠替代Spring的applicationContext.xml文件,并且被@Configuration注解標注的類,能夠自動注冊到IOC容器并進行實例化。
源碼詳見:org.springframework.context.annotation.Configuration。
@Target(ElementType.TYPE) @Retention(RetentionPolicy.RUNTIME) @Documented @Component public @interface Configuration { @AliasFor(annotation = Component.class) String value() default ""; // Since: 5.2 boolean proxyBeanMethods() default true; // Since: 6.0 boolean enforceUniqueMethods() default true; }
@Configuration注解中每個屬性的含義如下所示:
value:存入到Spring IOC容器中的Bean的id。
proxyBeanMethods:從Spring 5.2版本開始加入到@Configuration注解,表示被@Configuration注解標注的配置類是否會被代理,并且在配置類中使用@Bean注解生成的Bean對象在IOC容器中是否是單例對象,取值為true或者false。當取值為true時,表示full(全局)模式,此模式下被@Configuration注解標注的配置類會被代理,在配置類中使用@Bean注解注入到IOC容器中的Bean對象是單例模式,無論調用多少次被@Bean注解標注的方法,返回的都是同一個Bean對象。當取值為false時,表示lite(輕量級)模式,此模式下被@Configuration注解標注的配置類不會被代理,在配置類中使用@Bean注解注入到IOC容器中的Bean對象不是單例模式,每次調用被@Bean注解標注的方法時,都會返回一個新的Bean對象。默認的取值為true。
enforceUniqueMethods:從Spring 6.0開始加入到@Configuration注解,指定使用@Bean注解標注的方法是否需要具有唯一的方法名稱,取值為true或者false。當取值為true時,表示使用@Bean注解標注的方法具有唯一的方法名稱,并且這些方法名稱不會重疊。當取值為false時,表示使用@Bean注解標注的方法名稱不唯一,存在被重疊的風險。默認取值為true。
從@Configuration注解的源碼也可以看出,@Configuration注解本質上是一個@Component注解,所以,被@Configuration注解標注的配置類本身也會被注冊到IOC容器中。同時,@Configuration注解也會被@ComponentScan注解掃描到。
基于Spring的注解開發應用程序時,可以將@Configuration注解標注到某個類上。當某個類被@Configuration注解標注時,說明這個類是配置類,可以在這個類中使用@Bean注解向IOC容器中注入Bean對象,也可以使用@Autowired、@Inject和@Resource等注解來注入所需的Bean對象。
注意:基于Spring的注解模式開發應用程序時,在使用AnnotationConfigApplicationContext類創建IOC容器時,需要注意如下事項:
(1)如果調用的是AnnotationConfigApplicationContext類中傳入Class類型可變參數的構造方法來創建IOC容器,表示傳入使用@Configuration注解標注的配置類的Class對象來創建IOC容器,則標注到配置類上的@Configuration注解可以省略。
如果傳入的配置類上省略了@Configuration注解,則每次調用配置類中被@Bean注解標注的方法時,都會返回不同的Bean實例對象。
AnnotationConfigApplicationContext類中傳入Class類型可變參數的構造方法源碼如下所示:
public AnnotationConfigApplicationContext(Class<?>... componentClasses) { this(); register(componentClasses); refresh(); }
(2)如果調用的是AnnotationConfigApplicationContext類中傳入String類型可變參數的構造方法來創建IOC容器,表示傳入應用程序的包名來創建IOC容器,則標注到配置類上的@Configuration注解不能省略。
AnnotationConfigApplicationContext類中傳入String類型可變參數的構造方法源碼如下所示:
public AnnotationConfigApplicationContext(String... basePackages) { this(); scan(basePackages); refresh(); }
proxyBeanMethods屬性可取值為true或者false。取值為true時,無論調用多少次在被@Configuration注解標注的類中被@Bean注解標注的方法,返回的都是同一個Bean對象。取值為false時,每次調用在被@Configuration注解標注的類中被@Bean注解標注的方法,都回返回不同的Bean對象。
1.1 驗證proxyBeanMethods取值為true的情況
(1)創建Person類
Person類主要是用來注冊到IOC容器中,并實例化對象。
public class Person { private String name; public String getName() { return name; } public void setName(String name) { this.name = name; } }
(2)創建ConfigurationAnnotationConfig類
ConfigurationAnnotationConfig類的作用就是充當程序啟動的配置類,會在ConfigurationAnnotationConfig類上標注@Configuration注解,說明ConfigurationAnnotationConfig類是Spring啟動時的配置類。
@Configuration public class ConfigurationAnnotationConfig { @Bean public Person person(){ return new Person(); } }
可以看到,在ConfigurationAnnotationConfig類上標注了@Configuration注解,由于@Configuration注解中的proxyBeanMethods屬性默認為true,所以在ConfigurationAnnotationConfig類上的@Configuration注解省略了proxyBeanMethods屬性。
(3)創建ConfigurationAnnotationTest類
ConfigurationAnnotationTest類的作用就是整個案例程序的啟動類,對整個案例程序進行測試。
public class ConfigurationAnnotationTest { private static final Logger LOGGER = LoggerFactory.getLogger(ConfigurationAnnotationTest.class); public static void main(String[] args) { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(ConfigurationAnnotationConfig.class); ConfigurationAnnotationConfig config = context.getBean(ConfigurationAnnotationConfig.class); Person person1 = config.person(); Person person2 = config.person(); LOGGER.info("person1 == person2 ===>> {}", (person1 == person2)); } }
可以看到,在ConfigurationAnnotationTest類的main()方法中,首先基于AnnotationConfigApplicationContext常見了IOC容器context,從context中獲取了ConfigurationAnnotationConfig類的Bean實例對象config,接下來,調用兩次config的person()方法分別賦值給Person類型的局部變量person1和person2,最后打印person1是否等于person2的日志。
(4)測試案例
運行ConfigurationAnnotationTest類的main()方法,輸出的結果信息如下所示。
person1 是否等于 person2 ===>> true
通過輸出的結果信息可以看出,person1是否等于person2輸出的結果為true。說明當@Configuration注解中的proxyBeanMethods屬性為true時,每次調用使用@Configuration注解標注的類中被@Bean注解標注的方法時,都會返回同一個Bean實例對象。
1.2 驗證proxyBeanMethods取值為false的情況
驗證@Configuration注解中的proxyBeanMethods屬性為false的情況,與驗證proxyBeanMethods屬性為true的情況的案例程序基本一致,只是將ConfigurationAnnotationConfig類上標注的@Configuration注解的proxyBeanMethods屬性設置為false,案例實現的具體步驟如下所示。
(1)修改proxyBeanMethods屬性的值
@Configuration(proxyBeanMethods = false) public class ConfigurationAnnotationConfig { @Bean public Person person(){ return new Person(); } }
(2)測試案例
運行ConfigurationAnnotationTest類的main()方法,輸出的結果信息如下所示。
person1 是否等于 person2 ===>> false
從輸出的結果信息可以看出,person1是否等于person2輸出的結果為false。說明當@Configuration注解中的proxyBeanMethods屬性為false時,每次調用使用@Configuration注解標注的類中被@Bean注解標注的方法時,都會返回不同的Bean實例對象。
調用AnnotationConfigApplicationContext類的構造方法傳入配置類的Class對象創建IOC容器時,可以省略配置類上的@Configuration注解,案例的具體實現步驟如下所示。
(1)刪除@Configuration注解
public class ConfigurationAnnotationConfig { @Bean public Person person(){ return new Person(); } }
(2)測試案例
運行ConfigurationAnnotationTest類的main()方法,輸出的結果信息如下所示。
person1 是否等于 person2 ===>> false
從輸出的結果信息可以看到,輸出了person1是否等于person2的結果為false。說明調用AnnotationConfigApplicationContext類的構造方法傳入配置類的Class對象創建IOC容器時,可以省略配置類上的@Configuration注解,此時每次調用配置類中被@Bean注解標注的方法時,都會返回不同的Bean實例對象。
調用AnnotationConfigApplicationContext類的構造方法傳入包名創建IOC容器時,不能省略配置類上的@Configuration注解,案例的具體實現步驟如下所示。
(1)修改測試類
修改ConfigurationAnnotationTest類的main()方法中,創建AnnotationConfigApplicationContext對象的代碼,將調用傳入Class對象的構造方法修改為調用傳入String對象的方法,修改后的代碼如下所示。
public class ConfigurationAnnotationTest { private static final Logger LOGGER = LoggerFactory.getLogger(ConfigurationAnnotationTest.class); public static void main(String[] args) { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext("com.lwk.demo.spring.annocation.configuration.config"); ConfigurationAnnotationConfig config = context.getBean(ConfigurationAnnotationConfig.class); Person person1 = config.person(); Person person2 = config.person(); LOGGER.info("person1 是否等于 person2 ===>> {}", (person1 == person2)); } }
(2)刪除@Configuration注解
public class ConfigurationAnnotationConfig { @Bean public Person person(){ return new Person(); } }
(3)測試案例
運行ConfigurationAnnotationTest類的main()方法,可以看到程序拋出了異常信息,如下所示。
Exception in thread "main" org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'io.binghe.spring.annotation.chapter01.configuration.config.ConfigurationAnnotationConfig' available
從輸出的結果信息可以看出,調用AnnotationConfigApplicationContext類的構造方法傳入包名創建IOC容器時,不能省略配置類上的@Configuration注解,否則會拋出NoSuchBeanDefinitionException。
(4)添加@Configuration注解
@Configuration public class ConfigurationAnnotationConfig { @Bean public Person person(){ return new Person(); } }
(5)再次測試案例
再次運行ConfigurationAnnotationTest類的main()方法,輸出的結果信息如下所示。
person1 是否等于 person2 ===>> true
從輸出的結果信息可以看到,輸出了person1是否等于person2的結果為true,再次說明調用AnnotationConfigApplicationContext類的構造方法傳入包名創建IOC容器時,不能省略配置類上的@Configuration注解。
感謝各位的閱讀,以上就是“java中@Configuration使用場景是什么”的內容了,經過本文的學習后,相信大家對java中@Configuration使用場景是什么這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。