摘要:從功能上來說,沒有任何的毛病。解決方案每次接受請(qǐng)求之后,根據(jù)的不同,來執(zhí)行不同的業(yè)務(wù)邏輯。具體實(shí)現(xiàn)注解表示是個(gè)方法注解。通過的接口可以看到具體的使用方式,類上面使用注解,方法上使用注解,注解中傳入類中定義的名字即可。
業(yè)務(wù)場景
在與倉庫系統(tǒng)的對(duì)接過程中,我們使用了阿里巴巴的奇門規(guī)范。該規(guī)范中根據(jù)不同的method參數(shù)來確定不同的業(yè)務(wù),比如:
# 入庫單創(chuàng)建 method=taobao.qimen.entryorder.create # 庫存查詢 method=taobao.qimen.inventory.query # 商品同步接口 method=taobao.qimen.singleitem.synchronize
那么我們?cè)诮馕龅臅r(shí)候,常用的方式就是使用switch或者if來處理,以switch為例,實(shí)現(xiàn)代碼如下:
switch (method) { case "taobao.qimen.entryorder.create": return entryorderCreate(); case ""taobao.qimen.inventory.query: return inventoryQuery(); case "taobao.qimen.singleitem.synchronize": return singleitemSyncronize(); default: return ""; }
通過switch,我們根據(jù)不同的method能夠返回不同的執(zhí)行邏輯結(jié)果。從功能上來說,沒有任何的毛病。但是作為一個(gè)程序員,如果只是為了完成功能而寫代碼,那這樣的程序員是沒有靈魂的。
問題在奇門api技術(shù)文檔中,大概有50多個(gè)不同的業(yè)務(wù)接口method,這也就意味著我們至少要case 50次以上。你覺得一個(gè)switch中case 50次合理嗎?答案當(dāng)然是不合理的。
在這了再分享一句話:
任何一個(gè)傻瓜都能寫出計(jì)算機(jī)能理解的程序,而優(yōu)秀的程序員卻能寫出別人能讀得懂的程序。—— Martin Fowler解決方案
每次接受請(qǐng)求之后,根據(jù)method的不同,來執(zhí)行不同的業(yè)務(wù)邏輯。那么我們能不能將請(qǐng)求的method和需要執(zhí)行的業(yè)務(wù)邏輯方法做一個(gè)映射,這樣我們根據(jù)method就能直接找到具體的業(yè)務(wù)邏輯處理方法。
那么我們的method怎么和我們的業(yè)務(wù)方法映射綁定呢?解決方法是在每個(gè)業(yè)務(wù)方法上面增加一個(gè)注解(比如@Name)。那么問題來了,我們什么時(shí)候生成這樣的映射關(guān)系呢?
我們可以在容器啟動(dòng)的時(shí)候,就去生成這樣的映射關(guān)系。那么我們?cè)趺粗滥男╊惏司哂蠤Name注解的方法呢?
為了能快速獲取到包含@Name注解方法的類,我們?cè)黾右粋€(gè)類注解@MethodHandler,在方法上使用了@Name注解的類上我們加上一個(gè)@MethodHandler注解,這樣我們就能快速找到這樣的類。
@Name注解
@Target(ElementType.METHOD) @Documented @Retention(RetentionPolicy.RUNTIME) public @interface Name { String[] value() default {}; }
@Target(ElementType.METHOD)表示@Name是個(gè)方法注解。同時(shí)里面的value是個(gè)數(shù)組,是因?yàn)榭赡艽嬖诙鄠€(gè)method執(zhí)行相同業(yè)務(wù)邏輯的情況
@MethodHandler注解
@Target({ElementType.TYPE}) @Retention(RetentionPolicy.RUNTIME) @Documented public @interface MethodHandler { }
@Target({ElementType.TYPE})表示@MethodHandler是個(gè)類或者接口注解,此注解的作用是讓我們能快速找到包含@Name注解的方法。
MethodMappering
public class MethodMapping { //方法注解對(duì)應(yīng)的名字 public String[] names; //具體的執(zhí)行方法 public Method method; public MethodMapping(String[] names, Method method) { this.names = names; this.method = method; } }
這個(gè)類主要存儲(chǔ)奇門method和具體執(zhí)行的方法的映射
MethodNames
public class MethodNames { public static final String deliveryorder_confirm = "deliveryorder.confirm"; public static final String taobao_qimen_deliveryorder_confirm = "taobao.qimen.deliveryorder.confirm"; public static final String deliveryorder_batchconfirm = "deliveryorder.batchconfirm"; public static final String taobao_qimen_deliveryorder_batchconfirm = "taobao.qimen.deliveryorder.batchconfirm"; public static final String stockchange_report = "stockchange.report"; public static final String taobao_qimen_stockchange_report = "taobao.qimen.stockchange.report"; public static final String stockout_confirm = "stockout.confirm"; public static final String taobao_qimen_stockout_confirm = "taobao.qimen.stockout.confirm"; public static final String entryorder_confirm = "entryorder.confirm"; public static final String taobao_qimen_entryorder_confirm = "taobao.qimen.entryorder.confirm"; public static final String itemlack_report = "itemlack.report"; public static final String taobao_qimen_itemlack_report = "taobao.qimen.itemlack.report"; public static final String orderprocess_report = "orderprocess.report"; public static final String taobao_qimen_orderprocess_report = "taobao.qimen.orderprocess.report"; public static final String returnorder_confirm = "returnorder.confirm"; public static final String taobao_qimen_returnorder_confirm = "taobao.qimen.returnorder.confirm"; public static final String returnapply_report = "returnapply.report"; public static final String taobao_qimen_returnapply_report = "taobao.qimen.returnapply.report"; public static final String qimen_taobao_qianniu_cloudkefu_address_self_modify = "qimen.taobao.qianniu.cloudkefu.address.self.modify"; }
MethodNames類主要記錄了奇門中所有的method(此處只展示部分)
注解解析和檢查類DetectMethodAnnotation
@Component public class DetectMethodAnnotation extends AbstractReturner implements ApplicationContextAware, InitializingBean { private ApplicationContext applicationContext; //存儲(chǔ)類-方法 private HashMap> classMethodMap = new HashMap<>(); /** * 初始化容器后解析所有包含MethodHandler注解的類中包含Name注解的方法 * * @throws Exception */ @Override public void afterPropertiesSet() throws Exception { //獲取包含注解MethodHandler的類 Map methodHandlerMap = applicationContext.getBeansWithAnnotation(MethodHandler.class); methodHandlerMap.forEach((k, v) -> { Class> clazz = v.getClass(); Method[] methods = clazz.getDeclaredMethods();//獲取所有的方法 List methodMappings = new ArrayList<>(); for (Method method : methods) { //只解析@Name注解的,并且返回值為Returner的方法,方便對(duì)結(jié)果進(jìn)行解析 if (method.isAnnotationPresent(Name.class) && (method.getReturnType() == Returner.class)) { Name nameAnnotation = method.getAnnotation(Name.class); methodMappings.add(new MethodMapping(nameAnnotation.value(), method)); } } if (!methodMappings.isEmpty()) { classMethodMap.put(clazz.getName(), methodMappings); } }); } /** * 執(zhí)行 * * @param name * @return */ public Returner execute(String name, Object... parameters) throws Exception { if (!classMethodMap.containsKey(this.getClass().getName())) { return fail("類[" + this.getClass().getName() + "]未使用注解@MethodHandler注冊(cè)或未發(fā)現(xiàn)任何使用@Name注解的非繼承方法"); } List methodMappings = classMethodMap.get(this.getClass().getName()); for (MethodMapping methodMapping : methodMappings) { String[] names = methodMapping.names; if (Arrays.asList(names).contains(name)) { return (Returner) methodMapping.method.invoke(this, parameters); } } return fail("未發(fā)現(xiàn)使用注解 @Name("" + name + "") 為的方法"); } @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { this.applicationContext = applicationContext; } }
DetectMethodAnnotation的作用如下:
實(shí)現(xiàn)ApplicationContextAware接口,這樣能獲取到上下文對(duì)象ApplicationContext
實(shí)現(xiàn)InitializingBean接口的afterPropertiesSet()方法,此方法在容器啟動(dòng)之后只執(zhí)行一次,在此方法中可以解析所有的@Name注解
解析的數(shù)據(jù)存放在classMethodMap中,classMethodMap的數(shù)據(jù)結(jié)構(gòu)Hash<類名,List
提供一個(gè)方法execute,外部只需要傳遞method和業(yè)務(wù)邏輯方法的參數(shù)即可。
QimenController
@Controller @MethodHandler public class QimenController extends DetectMethodAnnotation { @Name({MethodNames.deliveryorder_confirm, MethodNames.taobao_qimen_deliveryorder_confirm}) public ReturnerdeliveryorderConfirm(String deliveryOrderCode) { logger.info("execute deliveryorderConfirm method with value " + deliveryOrderCode); return success(""); } @Name(MethodNames.stockchange_report) public Returner stockchangeReport() { return success(""); } }
通過QimenController的接口可以看到具體的使用方式,類上面使用@MethodHandler注解,方法上使用@Name注解,@Name注解中傳入MethodNames類中定義的名字即可。
測(cè)試
public class Run { public static final Logger logger = LoggerFactory.getLogger(Run.class); public static void main(String[] args) throws Exception { AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(Run.class); QimenController qimenController = applicationContext.getBean(QimenController.class); Returnerexecute = qimenController.execute(MethodNames.deliveryorder_confirm, "T123456789"); logger.info("deliveryorder_confirm:{}", execute); logger.info("stockchange_report:{}", qimenController.execute(MethodNames.stockchange_report)); applicationContext.close(); } }
執(zhí)行結(jié)果如下
[main] INFO solution.swithCase.QimenController - [18] - execute deliveryorderConfirm method with value T123456789 [main] INFO solution.swithCase.Run - [29] - deliveryorder_confirm:Returner(code=0, desc=null, body=) [main] INFO solution.swithCase.Run - [30] - stockchange_report:Returner(code=0, desc=null, body=)
Returner對(duì)象
@Data public class Returnerimplements Serializable { private String code; private String desc; private T body; }
此對(duì)象主要為了統(tǒng)一返回值,方便解析
總結(jié)首先要先明白解決方案思路才能理解代碼,其實(shí)就是把類-method-業(yè)務(wù)邏輯做一個(gè)映射,這樣就能直接通過接口中傳遞的method來找到具體的業(yè)務(wù)邏輯代碼。如果有不明白的地方可以在下面留言。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/75268.html
摘要:結(jié)構(gòu)型模式適配器模式橋接模式裝飾模式組合模式外觀模式享元模式代理模式。行為型模式模版方法模式命令模式迭代器模式觀察者模式中介者模式備忘錄模式解釋器模式模式狀態(tài)模式策略模式職責(zé)鏈模式責(zé)任鏈模式訪問者模式。 主要版本 更新時(shí)間 備注 v1.0 2015-08-01 首次發(fā)布 v1.1 2018-03-12 增加新技術(shù)知識(shí)、完善知識(shí)體系 v2.0 2019-02-19 結(jié)構(gòu)...
摘要:下一代服務(wù)端開發(fā)下一代服務(wù)端開發(fā)第部門快速開始第章快速開始環(huán)境準(zhǔn)備,,快速上手實(shí)現(xiàn)一個(gè)第章企業(yè)級(jí)服務(wù)開發(fā)從到語言的缺點(diǎn)發(fā)展歷程的缺點(diǎn)為什么是產(chǎn)生的背景解決了哪些問題為什么是的發(fā)展歷程容器的配置地獄是什么從到下一代企業(yè)級(jí)服務(wù)開發(fā)在移動(dòng)開發(fā)領(lǐng)域 《 Kotlin + Spring Boot : 下一代 Java 服務(wù)端開發(fā) 》 Kotlin + Spring Boot : 下一代 Java...
摘要:從使用到原理學(xué)習(xí)線程池關(guān)于線程池的使用,及原理分析分析角度新穎面向切面編程的基本用法基于注解的實(shí)現(xiàn)在軟件開發(fā)中,分散于應(yīng)用中多出的功能被稱為橫切關(guān)注點(diǎn)如事務(wù)安全緩存等。 Java 程序媛手把手教你設(shè)計(jì)模式中的撩妹神技 -- 上篇 遇一人白首,擇一城終老,是多么美好的人生境界,她和他歷經(jīng)風(fēng)雨慢慢變老,回首走過的點(diǎn)點(diǎn)滴滴,依然清楚的記得當(dāng)初愛情萌芽的模樣…… Java 進(jìn)階面試問題列表 -...
摘要:使用的好處知乎的回答不用自己組裝,拿來就用。統(tǒng)一配置,便于修改。 前言 只有光頭才能變強(qiáng) 回顧前面: 給女朋友講解什么是代理模式 包裝模式就是這么簡單啦 單例模式你會(huì)幾種寫法? 工廠模式理解了沒有? 在刷Spring書籍的時(shí)候花了點(diǎn)時(shí)間去學(xué)習(xí)了單例模式和工廠模式,總的來說還是非常值得的! 本來想的是刷完《Spring 實(shí)戰(zhàn) (第4版)》和《精通Spring4.x 企業(yè)應(yīng)用開發(fā)實(shí)戰(zhàn)》...
閱讀 2839·2021-09-28 09:45
閱讀 1511·2021-09-26 10:13
閱讀 908·2021-09-04 16:45
閱讀 3669·2021-08-18 10:21
閱讀 1094·2019-08-29 15:07
閱讀 2638·2019-08-29 14:10
閱讀 3151·2019-08-29 13:02
閱讀 2468·2019-08-29 12:31