<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
SpringBoot中通過@ConfigurationProperties
或@Value
註解就可以獲取組態檔中的屬性定義並繫結到Java Bean或屬性上,這也是我們平常使用最多的一種方式。但是小胖在開發過程中就遇到一個問題:在做MQ的開發中,組態檔中會設定多個生產者分別提供不同的業務能力,如果通過@ConfigurationProperties
註解來實現的話,這就意味著需要建立多個屬性一樣的設定類,雖然說可以實現功能,但是很明顯,這不是一個很好的設計。場景如下所示:
producer1: password: xxx app: xxx address: url1 enabled: false producer2: password: xxx app: xxx address: url1 enabled: false
在我們日常的開發工作中,經常可以見到的是通過自定義註解+攔截器+反射從而實現對許可權的校驗或者對實體類欄位值格式進行校驗。那麼,我們是不是也可以參考這個思路達到我們的目的呢?答案是肯定的,其實如果對Mabatis等元件比較熟悉的話,就可以看到這樣的設計。我們話不多少,開搞~
以下內容,為了方便,我們將設定相關內容改為人員(people)
首先,有一點是不會改變的,我們需要自定義一個設定類,用於讀取組態檔中的設定。這裡,我們需要改變一下我們組態檔資訊裡。將所有的設定資訊放到一個類裡。
my: peoples: people1: userName: 張三 userSex: 男 people2: userName: 李四 userSex: 女
然後,定義一個設定類用來接收,這裡通過@ConfigurationProperties
註解實現對設定的注入。要注意,因為我們在peoples下面有很多的people,因此,屬性應給定義的是一個MAP的型別。
@Component @ConfigurationProperties(prefix = "my",ignoreUnknownFields = false) public class PeopleConfigs { private Map<String, PeopleEntity> peoples; public Map<String, PeopleEntity> getPeoples() { return peoples; } public void setPeoples(Map<String, PeopleEntity> peoples) { this.peoples = peoples; } @Override public String toString() { return "PeopleConfigs{" + "peoples=" + peoples + '}'; } } public class PeopleEntity { private String userName; private String userSex; public String getUserName() { return userName; } public void setUserName(String userName) { this.userName = userName; } public String getUserSex() { return userSex; } public void setUserSex(String userSex) { this.userSex = userSex; } @Override public String toString() { return "PeopleEntity{" + "userName='" + userName + ''' + ", userSex='" + userSex + ''' + '}'; } }
這樣,Springboot就會自動載入我們這個設定類。但是,這個的整個PeopleConfigs
是一個Bean,並不能達到我們本文的目的,因此我們進行後續的步驟。
我們宣告一個執行時的註解,在屬性上進行使用。這裡定義name用來標記需要注入的是哪個人。
@Retention(RetentionPolicy.RUNTIME) @Target({ElementType.FIELD}) public @interface People { String name() default ""; }
首先,定義一個autoConfig的設定類,該類通過@EnableConfigurationProperties
註解,指定PeopleConfig Bean在本類之前進行裝載。通過@Bean
方法註解進行bean宣告,此處呼叫的是單個people設定類的bean生成的方法。
@Configuration @EnableConfigurationProperties({PeopleConfigs.class}) public class PeopleAutoConfig { @Autowired PeopleConfigs peopleConfigs; @Bean public PeopleRegister peopleRegister(){ return new PeopleRegister(peopleConfigs); } }
這裡不得不提到BeanPostProcessor
類,該類為我們提供了springBoot在bean初始化前後方便我們進行其他自定義操作的一些介面。我們這裡通過實現postProcessBeforeInitialization
方法,在bean裝載之前,通過反射判斷對應bean上是否有我們自定義的people註解。如果有,則進行注入操作。詳細程式碼如下:
public class PeopleRegister implements BeanPostProcessor, ApplicationContextAware { private final PeopleConfigs peopleConfigs; private GenericApplicationContext applicationContext; PeopleRegister(PeopleConfigs peopleConfigs){ this.peopleConfigs = peopleConfigs; } @Override public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException { Class<?> beanClass = AopUtils.getTargetClass(bean); Field[] fields = beanClass.getDeclaredFields(); Field[] var5 = fields; int var6 = fields.length; for(int var7 = 0;var7<var6;var7++){ Field field = var5[var7]; People annotation = field.getAnnotation(People.class); if (annotation!=null){ PeopleEntity entity = this.peopleConfigs.getPeoples().get(annotation.name()); if (!this.applicationContext.containsBean(annotation.name())){ ConfigurableListableBeanFactory beanFactory = this.applicationContext.getBeanFactory(); Object wrapperBean = beanFactory.initializeBean(entity, annotation.name()); beanFactory.registerSingleton(annotation.name(), Objects.requireNonNull(wrapperBean)); } try{ field.setAccessible(true); field.set(bean, this.applicationContext.getBean(annotation.name(), PeopleEntity.class)); }catch (Exception e){ e.printStackTrace(); } } } return bean; } @Override public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException { return bean; } @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { this.applicationContext = (GenericApplicationContext)applicationContext; } }
前面工作進行完成後,接下來就是我們的使用環節,這裡,我們僅需要通過@People(name = "人")
指定即可:
@Controller public class BaseController { @Autowired PeopleConfigs peopleConfigs; @People(name = "people1") PeopleEntity people1; @People(name = "people2") PeopleEntity people2; @ResponseBody @GetMapping("/test") public String test() { return peopleConfigs.toString()+people1.toString()+people2.toString(); } }
本文我們通過自定義註解+反射實現了設定類的自動注入,看完以後是不是覺得如此簡單。我們只是把發射注入的邏輯從攔截器換到了ostProcessBeforeInitialization中來而已。
在工作中,其實有很多業務場景可以讓我們去進行友好的設計,關鍵在於我們有沒有這個意識,大家一起進步。
有些JRM看到這個設計,是不是感覺也很想吐,明明能很簡單搞得卻搞得這麼麻煩,還寫註冊類什麼的。其實實際應用中,比如MQ一類的,我們在bean注入後,是希望直接啟動連線的,在銷燬時希望能自動關閉連線,而不是每次使用的時候都要自己對連線進行管理。因此才如此進行定義和設計。
到此這篇關於SpringBoot通過自定義註解實現設定類的自動注入的實現的文章就介紹到這了,更多相關SpringBoot設定類自動注入內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45