解決Spring Boot 多模塊注入訪問(wèn)不到j(luò)ar包中的Bean問(wèn)題
情景描述
一個(gè)聚合項(xiàng)目spring-security-tutorial,其中包括4個(gè)module,pom如下所示:
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>com.github.jdkong.security</groupId> <artifactId>spring-security-tutorial</artifactId> <version>1.0-SNAPSHOT</version> <packaging>pom</packaging> <modules>`` <module>security-core</module> <module>security-app</module> <module>security-browser</module> <module>security-demo</module> </modules> <!-- 其他部分省略--> </project>
在此項(xiàng)目中,子項(xiàng)目security-browser是一個(gè)簡(jiǎn)單的maven項(xiàng)目,打成jar包,供security-demo使用,security-demo項(xiàng)目是一個(gè)springboot項(xiàng)目。
問(wèn)題描述
在security-browser項(xiàng)目中自動(dòng)注入了一個(gè)配置類,如下所示:
/** * @author jdkong */ @Slf4j @Configuration public class BrowserSecurityConfig extends WebSecurityConfigurerAdapter { @Override protected void configure(HttpSecurity http) throws Exception { http.formLogin() .and() .authorizeRequests() .anyRequest() .authenticated(); } }
在security-demo中使用此配置類時(shí),不起作用。
問(wèn)題分析
導(dǎo)致此類問(wèn)題的主要原因是,此類不在Spring Boot的組件掃描范圍之內(nèi)。
1. 關(guān)于 Spring Boot 自動(dòng)注入及組件掃描
在平時(shí)使用 Spring Boot 時(shí),常常會(huì)使用到@Configuration,@Contoller,@Service,@Component等注解,被添加這些注解的類,在 Spring Boot 啟動(dòng)時(shí),會(huì)自動(dòng)被 Spring 容器管理起來(lái)。
上面提到了,添加了一些注解的類會(huì)在Spring Boot 容器啟動(dòng)時(shí),被加載到Spring 容器中。那么,組件掃描的作用就是:當(dāng) Spring Boot 啟動(dòng)時(shí),根據(jù)定義的掃描路徑,把符合掃描規(guī)則的類裝配到spring容器中。
2. Spring Boot 中 @ComponentScan
簡(jiǎn)單的介紹了@ComponentScan的基礎(chǔ)作用,這個(gè)注解為我們使用提供了一些可自定義配置屬性,先來(lái)看看@ComponentScan注解源碼:
@Retention(RetentionPolicy.RUNTIME) @Target({ElementType.TYPE}) @Documented @Repeatable(ComponentScans.class) public @interface ComponentScan { // 指定掃描包的位置(同:basePackages 屬性),可以是單個(gè)路徑,也可以是掃描的路徑數(shù)組 @AliasFor("basePackages") String[] value() default {}; // 指定掃描包的位置(同:value 屬性) @AliasFor("value") String[] basePackages() default {}; // 指定具體的掃描的類 Class<?>[] basePackageClasses() default {}; // bean的名稱的生成器 Class<? extends BeanNameGenerator> nameGenerator() default BeanNameGenerator.class; Class<? extends ScopeMetadataResolver> scopeResolver() default AnnotationScopeMetadataResolver.class; ScopedProxyMode scopedProxy() default ScopedProxyMode.DEFAULT; // 控制符合組件檢測(cè)條件的類文件 默認(rèn)是包掃描下的 **/*.class String resourcePattern() default "**/*.class"; // 是否開(kāi)啟對(duì)@Component,@Repository,@Service,@Controller的類進(jìn)行檢測(cè) boolean useDefaultFilters() default true; // 包含的過(guò)濾條件 // 1. FilterType.ANNOTATION: 按照注解過(guò)濾 // 2. FilterType.ASSIGNABLE_TYPE: 按照給定的類型 // 3. FilterType.ASPECTJ: 使用ASPECTJ表達(dá)式 // 4. FilterType.REGEX: 正則 // 5. FilterType.CUSTOM: 自定義規(guī)則 ComponentScan.Filter[] includeFilters() default {}; // 排除的過(guò)濾條件,用法和includeFilters一樣 ComponentScan.Filter[] excludeFilters() default {}; boolean lazyInit() default false; @Retention(RetentionPolicy.RUNTIME) @Target({}) public @interface Filter { FilterType type() default FilterType.ANNOTATION; @AliasFor("classes") Class<?>[] value() default {}; @AliasFor("value") Class<?>[] classes() default {}; String[] pattern() default {}; } }
總結(jié)一下@ComponentScan的常用方式如下:
通過(guò)使用value,basePackages屬性來(lái)指定掃描范圍;
自定掃描路徑下邊帶有@Controller,@Service,@Repository,@Component注解加入Spring容器
通過(guò)includeFilters加入掃描路徑下沒(méi)有以上注解的類加入spring容器
通過(guò)excludeFilters過(guò)濾出不用加入spring容器的類
自定義增加了@Component注解的注解方式
3. Spring Boot 中 @SpringBootApplication
在創(chuàng)建Spring Boot 項(xiàng)目之后,在默認(rèn)的啟動(dòng)類上會(huì)被添加@SpringBootApplication注解,這個(gè)注解默認(rèn)幫我們開(kāi)啟一些自動(dòng)配置的功能,比如:基于Java的Spring配置,組件掃描,特別是用于啟用Spring Boot的自動(dòng)配置功能。
@Target({ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Documented @Inherited @SpringBootConfiguration @EnableAutoConfiguration // 允許自動(dòng)配置 @ComponentScan( excludeFilters = {@Filter( // 定義排除規(guī)則 type = FilterType.CUSTOM, // 采用自定義的方式 classes = {TypeExcludeFilter.class} // 自定義實(shí)現(xiàn)邏輯 ), @Filter( // 同上 type = FilterType.CUSTOM, classes = {AutoConfigurationExcludeFilter.class} )} ) public @interface SpringBootApplication { // 為 @EnableAutoConfiguration 添加 exclude 規(guī)則 @AliasFor( annotation = EnableAutoConfiguration.class, attribute = "exclude" ) Class<?>[] exclude() default {}; // 為 @EnableAutoConfiguration 添加 excludeName 規(guī)則 @AliasFor( annotation = EnableAutoConfiguration.class, attribute = "excludeName" ) String[] excludeName() default {}; // 為 @ComponentScan 添加 basePackages 規(guī)則 @AliasFor( annotation = ComponentScan.class, attribute = "basePackages" ) String[] scanBasePackages() default {}; // 為 @ComponentScan 添加 basePackageClasses 規(guī)則 @AliasFor( annotation = ComponentScan.class, attribute = "basePackageClasses" ) Class<?>[] scanBasePackageClasses() default {}; }
從上面的源碼部分可以看到,@SpringBootApplication是一個(gè)組合注解,也就相當(dāng)于使用一個(gè)@SpringBootApplication可以替代@SpringBootConfiguration,@EnableAutoConfiguration,@ComponentScan幾個(gè)注解聯(lián)合使用。
注:此注釋從SpringBoot 1.2開(kāi)始提供,這意味著如果你運(yùn)行的是較低的版本,并且如果你需要這些功能,你需要手動(dòng)添加@Configuration,@CompnentScan和@EnableAutoConfiguration。
那么,可能會(huì)有這樣的問(wèn)題,我只是使用了一個(gè)@SpringBootApplication注解,但是我如何對(duì)@ComponentScan的屬性做自定義配置呢?
當(dāng)然,Spring 團(tuán)隊(duì)已經(jīng)很好的解決了這個(gè)問(wèn)題,在@SpringBootApplication注解類中的屬性上添加@AliasFor注解,從而實(shí)現(xiàn)通過(guò)對(duì)@SpringBootApplication中的屬性進(jìn)行自定義,達(dá)到對(duì)對(duì)應(yīng)的注解的屬性的自定義。
比如:
@AliasFor( annotation = ComponentScan.class, attribute = "basePackages" ) String[] scanBasePackages() default {};
這段代碼就是實(shí)現(xiàn),通過(guò)對(duì)@SpringBootApplication的屬性scanBasePackages,實(shí)現(xiàn)對(duì)@ComponentScan中的屬性basePackages進(jìn)行自定義。
4. 回答開(kāi)篇問(wèn)題
先看看項(xiàng)目結(jié)構(gòu),項(xiàng)目入口文件在子項(xiàng)目security-demo中,并且入口類所在包位置為:package com.github.jdkong.security。
也就是說(shuō),在不做任何配置的情況下,此項(xiàng)目只會(huì)掃描當(dāng)前包路徑及其子路徑下的文件,并將符合條件的對(duì)象注入到容器中管理。
再看看配置文件所在的包路徑位置:package com.github.jdkong.browser.config,可見(jiàn)此包路徑并不在項(xiàng)目掃描的路徑范圍之內(nèi)。
這也就導(dǎo)致了,我們定義的配置類,雖然加了@Configuration也不會(huì)對(duì)我們的項(xiàng)目起到作用。
可以對(duì)項(xiàng)目注解進(jìn)行稍微修改,制定掃描包的范圍,就可以簡(jiǎn)單的解決這個(gè)問(wèn)題。如下:
@SpringBootApplication(scanBasePackages="com.github.jdkong") public class SecurityApplication { public static void main(String[] args) { SpringApplication.run(SecurityApplication.class,args); } }
5. 補(bǔ)充說(shuō)明:@AliasFor
在Spring注解中,經(jīng)常會(huì)發(fā)現(xiàn)很多注解的不同屬性起著相同的作用,比如@ComponentScan的value屬性和basePackages屬性。所以在使用的時(shí)候就需要做一些基本的限制,比如value和basePackages的值不能沖突,比如任意設(shè)置value或者設(shè)置basePackages屬性的值,都能夠通過(guò)另一個(gè)屬性來(lái)獲取值等等。為了統(tǒng)一處理這些情況,Spring創(chuàng)建了@AliasFor標(biāo)簽。
以上這篇解決Spring Boot 多模塊注入訪問(wèn)不到j(luò)ar包中的Bean問(wèn)題就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot構(gòu)建Restful service完成Get和Post請(qǐng)求
這篇文章主要介紹了SpringBoot構(gòu)建Restful service完成Get和Post請(qǐng)求的示例代碼,感興趣的朋友一起看看吧2017-08-08java實(shí)現(xiàn)HttpClient異步請(qǐng)求資源的方法
這篇文章主要介紹了java實(shí)現(xiàn)HttpClient異步請(qǐng)求資源的方法,實(shí)例分析了java基于http協(xié)議實(shí)現(xiàn)異步請(qǐng)求的技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-07-07SpringBoot項(xiàng)目加入沖突動(dòng)態(tài)監(jiān)測(cè)算法的實(shí)現(xiàn)
沖突動(dòng)態(tài)監(jiān)測(cè)算法是一種網(wǎng)絡(luò)通信中的沖突檢測(cè)方法,適用于無(wú)線網(wǎng)絡(luò)或其他共享傳輸介質(zhì)的環(huán)境,本文主要介紹了SpringBoot項(xiàng)目加入沖突動(dòng)態(tài)監(jiān)測(cè)算法的實(shí)現(xiàn),感興趣的可以了解一下2023-09-09Java實(shí)現(xiàn)對(duì)象轉(zhuǎn)CSV格式
CSV是一種逗號(hào)分隔值格式的文件,一般用來(lái)存儲(chǔ)數(shù)據(jù)的純文本格式文件。Java對(duì)象轉(zhuǎn)CSV,有現(xiàn)成的工具包,commons-lang3 的ReflectionToStringBuilder 就可以簡(jiǎn)單的解決的對(duì)象轉(zhuǎn)CSV,快跟隨小編一起學(xué)習(xí)一下吧2022-06-06SpringCloud hystrix服務(wù)降級(jí)概念介紹
什么是服務(wù)降級(jí)?當(dāng)服務(wù)器壓力劇增的情況下,根據(jù)實(shí)際業(yè)務(wù)情況及流量,對(duì)一些服務(wù)和頁(yè)面有策略的不處理或換種簡(jiǎn)單的方式處理,從而釋放服務(wù)器資源以保證核心交易正常運(yùn)作或高效運(yùn)作2022-09-09spring security在分布式項(xiàng)目下的配置方法(案例詳解)
這篇文章主要介紹了spring security在分布式項(xiàng)目下的配置方法,本文通過(guò)一個(gè)項(xiàng)目案例給大家詳細(xì)介紹,通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-10-10java中基本注解的知識(shí)點(diǎn)總結(jié)
在本篇文章里小編給大家整理的是一篇關(guān)于java中基本注解的知識(shí)點(diǎn)總結(jié),有需要的朋友們可以跟著學(xué)習(xí)下。2021-06-06