干貨點
創(chuàng)新互聯(lián)建站服務(wù)項目包括萬秀網(wǎng)站建設(shè)、萬秀網(wǎng)站制作、萬秀網(wǎng)頁制作以及萬秀網(wǎng)絡(luò)營銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢、行業(yè)經(jīng)驗、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,萬秀網(wǎng)站推廣取得了明顯的社會效益與經(jīng)濟效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到萬秀省份的部分城市,未來相信會繼續(xù)擴大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!
通過閱讀該篇博客,你可以了解了解java的反射機制、可以了解如何基于spring生命周期使用自定義注解解決日常研發(fā)問題。具體源碼可以點擊鏈接。
問題描述
在日常研發(fā)中,經(jīng)常會遇見業(yè)務(wù)A的某個action被觸發(fā)后,同時觸發(fā)業(yè)務(wù)B的action的行為,這種單對單的形式可以直接在業(yè)務(wù)A的action執(zhí)行結(jié)束后直接調(diào)用業(yè)務(wù)B的action,那么如果是單對多的情況呢?
方案解決
這里提供一種在日常研發(fā)中經(jīng)常使用到的機制,基于spring實現(xiàn)的事件驅(qū)動,即在業(yè)務(wù)A的action執(zhí)行完,拋出一個事件,而業(yè)務(wù)B、C、D等監(jiān)聽到該事件后處理相應(yīng)的業(yè)務(wù)。
場景范例
這里提供一個場景范例,該范例基于springboot空殼項目實現(xiàn),具體可以查看源碼,此處只梳理關(guān)鍵步驟。
步驟一:
定義一個注解,標(biāo)志接收事件的注解,即所有使用了該注解的函數(shù)都會在對應(yīng)事件被拋出的時候被調(diào)用,該注解實現(xiàn)比較簡單,代碼如下
/** * @author xifanxiaxue * @date 3/31/19 * @desc 接收事件的注解 */ @Documented @Retention(RetentionPolicy.RUNTIME) @Target({ElementType.METHOD}) public @interface ReceiveAnno { // 監(jiān)聽的事件 Class clz(); }
如果想了解注解多個參數(shù)的意義是什么的可以點擊鏈接查看博主之前寫過文章。
定義事件接口
/** * @author xifanxiaxue * @date 3/31/19 * @desc */ public interface IEvent { }
所有事件都需要實現(xiàn)該接口,主要是為了后面泛型和類型識別。
定義MethodInfo
/** * @author xifanxiaxue * @date 3/31/19 * @desc */ public class MethodInfo { public Object obj; public Method method; public static MethodInfo valueOf(Method method, Object obj) { MethodInfo info = new MethodInfo(); info.method = method; info.obj = obj; return info; } public Object getObj() { return obj; } public Method getMethod() { return method; } }
該類只是做了Object和Method的封裝,沒有其他作用。
步驟二:
實現(xiàn)一個事件容器,該容器的作用是存放各個事件以及需要觸發(fā)的各個業(yè)務(wù)的method的對應(yīng)關(guān)系。
/** * @author xifanxiaxue * @date 3/31/19 * @desc 事件容器 */ public class EventContainer { private static Map, List > eventListMap = new HashMap<>(); public static void addEventToMap(Class clz, Method method, Object obj) { List methodInfos = eventListMap.get(clz); if (methodInfos == null) { methodInfos = new ArrayList<>(); eventListMap.put(clz, methodInfos); } methodInfos.add(MethodInfo.valueOf(method, obj)); } public static void submit(Class clz) { List methodInfos = eventListMap.get(clz); if (methodInfos == null) { return; } for (MethodInfo methodInfo : methodInfos) { Method method = methodInfo.getMethod(); try { method.setAccessible(true); method.invoke(methodInfo.getObj()); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); } } } }
其中的addEventToMap函數(shù)的作用是將對應(yīng)的事件、事件觸發(fā)后需要觸發(fā)的對應(yīng)業(yè)務(wù)內(nèi)的Method存放在eventListMap內(nèi);而submit函數(shù)會在其他業(yè)務(wù)類內(nèi)拋出事件的時候被調(diào)用,而作用是從eventListMap中取出對應(yīng)的Method,并通過反射觸發(fā)。
步驟三:
實現(xiàn)事件處理器,該事件處理器的作用是在bean被spring容器實例化后去判斷對應(yīng)的bean是否有相應(yīng)函數(shù)加了@ReceiveAnno注解,如果有則從中取出對應(yīng)的Event并放入EventContainer中。
/** * @author xifanxiaxue * @date 3/31/19 * @desc 事件處理器 */ @Component public class EventProcessor extends InstantiationAwareBeanPostProcessorAdapter { @Override public boolean postProcessAfterInstantiation(Object bean, String beanName) throws BeansException { ReflectionUtils.doWithLocalMethods(bean.getClass(), new ReflectionUtils.MethodCallback() { @Override public void doWith(Method method) throws IllegalArgumentException, IllegalAccessException { ReceiveAnno anno = method.getAnnotation(ReceiveAnno.class); if (anno == null) { return; } Class clz = anno.clz(); try { if (!IEvent.class.isInstance(clz.newInstance())) { FormattingTuple message = MessageFormatter.format("{}沒有實現(xiàn)IEvent接口", clz); throw new RuntimeException(message.getMessage()); } } catch (InstantiationException e) { e.printStackTrace(); } EventContainer.addEventToMap(clz, method, bean); } }); return super.postProcessAfterInstantiation(bean, beanName); } }
關(guān)于InstantiationAwareBeanPostProcessorAdapter的描述,有需要的可以查看我之前的文章,其中比較詳細(xì)描述到Spring中的InstantiationAwareBeanPostProcessor類的作用。
步驟四:
對應(yīng)的業(yè)務(wù)類的實現(xiàn)如下:
/** * @author xifanxiaxue * @date 3/31/19 * @desc */ @Slf4j @Service public class AFuncService implements IAFuncService { @Override public void login() { log.info("[{}]拋出登錄事件 ... ", this.getClass()); EventContainer.submit(LoginEvent.class); } }
A業(yè)務(wù)類,login會在被調(diào)用的生活拋出LoginEvent事件。
/** * @author xifanxiaxue * @date 3/31/19 * @desc */ @Service @Slf4j public class BFuncService implements IBFuncService { @ReceiveAnno(clz = LoginEvent.class) private void doAfterLogin() { log.info("[{}]監(jiān)聽到登錄事件 ... ", this.getClass()); } }
/** * @author xifanxiaxue * @date 3/31/19 * @desc */ @Service @Slf4j public class CFuncService implements ICFuncService { @ReceiveAnno(clz = LoginEvent.class) private void doAfterLogin() { log.info("[{}]監(jiān)聽到登錄事件 ... ", this.getClass()); } }
B和C業(yè)務(wù)類的doAfterLogin都分別加了注解 @ReceiveAnno(clz = LoginEvent.class) ,在監(jiān)聽到事件LoginEvent后被觸發(fā)。
為了觸發(fā)方便,我在spring提供的測試類內(nèi)加了實現(xiàn),代碼如下:
@RunWith(SpringRunner.class) @SpringBootTest public class EventMechanismApplicationTests { @Autowired private AFuncService aFuncService; @Test public void contextLoads() { aFuncService.login(); } }
可以從中看出啟動該測試類后,會調(diào)用業(yè)務(wù)A的login函數(shù),而我們要的效果是B業(yè)務(wù)類和C業(yè)務(wù)類的doAfterLogin函數(shù)會被自動觸發(fā),那么結(jié)果如何呢?
結(jié)果打印
我們可以從結(jié)果打印中看到,在業(yè)務(wù)類A的login函數(shù)觸發(fā)后,業(yè)務(wù)類B和業(yè)務(wù)類C都監(jiān)聽到了監(jiān)聽到登錄事件,證明該機制正常解決了單對多的行為觸發(fā)問題。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對創(chuàng)新互聯(lián)的支持。