干掉 if 语句,一个不留的那种!
?作者:烏塔卡
juejin.cn/post/6951764927958745124
業(yè)務(wù)場景
近日在公司領(lǐng)到一個小需求,需要對之前已有的試用用戶申請規(guī)則進行拓展。我們的場景大概如下所示:
if?(是否海外用戶)?{return?false; }if?(刷單用戶)?{return?false; }if?(未付費用戶?&&?不再服務(wù)時段)?{return?false }if?(轉(zhuǎn)介紹用戶?||?付費用戶?||?內(nèi)推用戶)?{return?true; }按照上述的條件我們可以得出的結(jié)論是:
咱們的的主要流程主要是基于 and 或者 or 的關(guān)系。
如果有一個不匹配的話,其實咱們后續(xù)的流程是不用執(zhí)行的,就是需要具備一個短路的功能。
對于目前的現(xiàn)狀來說,我如果在原有的基礎(chǔ)上來改,只要稍微注意一下解決需求不是很大的問題,但是說后面可維護性非常差。
后面進過權(quán)衡過后,我還是決定將這個部分進行重構(gòu)一下。
規(guī)則執(zhí)行器
針對這個需求,我首先梳理了一下咱們規(guī)則執(zhí)行器大概的設(shè)計, 然后我設(shè)計了一個 V1 版本和大家一起分享一下,如果大家也有這樣的 case 可以給我分享留言,下面部分主要是設(shè)計和實現(xiàn)的流程和 code.
規(guī)則執(zhí)行器的設(shè)計
對于規(guī)則的抽象并實現(xiàn)規(guī)則
//?業(yè)務(wù)數(shù)據(jù) @Data public?class?RuleDto?{private?String?address;private?int?age; }//?規(guī)則抽象 public?interface?BaseRule?{boolean?execute(RuleDto?dto); }//?規(guī)則模板 public?abstract?class?AbstractRule?implements?BaseRule?{protected?<T>?T?convert(RuleDto?dto)?{return?(T)?dto;}@Overridepublic?boolean?execute(RuleDto?dto)?{return?executeRule(convert(dto));}protected?<T>?boolean?executeRule(T?t)?{return?true;} }//?具體規(guī)則-?例子1 public?class?AddressRule?extends?AbstractRule?{@Overridepublic?boolean?execute(RuleDto?dto)?{System.out.println("AddressRule?invoke!");if?(dto.getAddress().startsWith(MATCH_ADDRESS_START))?{return?true;}return?false;} }//?具體規(guī)則-?例子2 public?class?NationalityRule?extends?AbstractRule?{@Overrideprotected?<T>?T?convert(RuleDto?dto)?{NationalityRuleDto?nationalityRuleDto?=?new?NationalityRuleDto();if?(dto.getAddress().startsWith(MATCH_ADDRESS_START))?{nationalityRuleDto.setNationality(MATCH_NATIONALITY_START);}return?(T)?nationalityRuleDto;}@Overrideprotected?<T>?boolean?executeRule(T?t)?{System.out.println("NationalityRule?invoke!");NationalityRuleDto?nationalityRuleDto?=?(NationalityRuleDto)?t;if?(nationalityRuleDto.getNationality().startsWith(MATCH_NATIONALITY_START))?{return?true;}return?false;} }//?常量定義 public?class?RuleConstant?{public?static?final?String?MATCH_ADDRESS_START=?"北京";public?static?final?String?MATCH_NATIONALITY_START=?"中國"; }執(zhí)行器構(gòu)建
public?class?RuleService?{private?Map<Integer,?List<BaseRule>>?hashMap?=?new?HashMap<>();private?static?final?int?AND?=?1;private?static?final?int?OR?=?0;public?static?RuleService?create()?{return?new?RuleService();}public?RuleService?and(List<BaseRule>?ruleList)?{hashMap.put(AND,?ruleList);return?this;}public?RuleService?or(List<BaseRule>?ruleList)?{hashMap.put(OR,?ruleList);return?this;}public?boolean?execute(RuleDto?dto)?{for?(Map.Entry<Integer,?List<BaseRule>>?item?:?hashMap.entrySet())?{List<BaseRule>?ruleList?=?item.getValue();switch?(item.getKey())?{case?AND://?如果是?and?關(guān)系,同步執(zhí)行System.out.println("execute?key?=?"?+?1);if?(!and(dto,?ruleList))?{return?false;}break;case?OR://?如果是?or?關(guān)系,并行執(zhí)行System.out.println("execute?key?=?"?+?0);if?(!or(dto,?ruleList))?{return?false;}break;default:break;}}return?true;}private?boolean?and(RuleDto?dto,?List<BaseRule>?ruleList)?{for?(BaseRule?rule?:?ruleList)?{boolean?execute?=?rule.execute(dto);if?(!execute)?{//?and?關(guān)系匹配失敗一次,返回?falsereturn?false;}}//?and?關(guān)系全部匹配成功,返回?truereturn?true;}private?boolean?or(RuleDto?dto,?List<BaseRule>?ruleList)?{for?(BaseRule?rule?:?ruleList)?{boolean?execute?=?rule.execute(dto);if?(execute)?{//?or?關(guān)系匹配到一個就返回?truereturn?true;}}//?or?關(guān)系一個都匹配不到就返回?falsereturn?false;} }執(zhí)行器的調(diào)用
public?class?RuleServiceTest?{@org.junit.Testpublic?void?execute()?{//規(guī)則執(zhí)行器//優(yōu)點:比較簡單,每個規(guī)則可以獨立,將規(guī)則,數(shù)據(jù),執(zhí)行器拆分出來,調(diào)用方比較規(guī)整//缺點:數(shù)據(jù)依賴公共傳輸對象 dto//1.?定義規(guī)則??init?ruleAgeRule?ageRule?=?new?AgeRule();NameRule?nameRule?=?new?NameRule();NationalityRule?nationalityRule?=?new?NationalityRule();AddressRule?addressRule?=?new?AddressRule();SubjectRule?subjectRule?=?new?SubjectRule();//2.?構(gòu)造需要的數(shù)據(jù)?create?dtoRuleDto?dto?=?new?RuleDto();dto.setAge(5);dto.setName("張三");dto.setAddress("北京");dto.setSubject("數(shù)學(xué)");;//3.?通過以鏈?zhǔn)秸{(diào)用構(gòu)建和執(zhí)行?rule?executeboolean?ruleResult?=?RuleService.create().and(Arrays.asList(nationalityRule,?nameRule,?addressRule)).or(Arrays.asList(ageRule,?subjectRule)).execute(dto);System.out.println("this?student?rule?execute?result?:"?+?ruleResult);} }總結(jié)
規(guī)則執(zhí)行器的優(yōu)點和缺點
優(yōu)點:
比較簡單,每個規(guī)則可以獨立,將規(guī)則,數(shù)據(jù),執(zhí)行器拆分出來,調(diào)用方比較規(guī)整;
我在 Rule 模板類中定義 convert 方法做參數(shù)的轉(zhuǎn)換這樣可以能夠,為特定 rule 需要的場景數(shù)據(jù)提供拓展。
缺點:
上下 rule 有數(shù)據(jù)依賴性,如果直接修改公共傳輸對象 dto 這樣設(shè)計不是很合理,建議提前構(gòu)建數(shù)據(jù)。
2021 最新版 Spring Boot 速記教程
2W 字你全面認識 Nginx
47K Star 的SpringBoot+MyBatis+docker電商項目,附帶超詳細的文檔!
寫博客能月入10K?
一款基于 Spring Boot 的現(xiàn)代化社區(qū)(論壇/問答/社交網(wǎng)絡(luò)/博客)
這或許是最美的Vue+Element開源后臺管理UI
推薦一款高顏值的 Spring Boot 快速開發(fā)框架
一款基于 Spring Boot 的現(xiàn)代化社區(qū)(論壇/問答/社交網(wǎng)絡(luò)/博客)
13K點贊都基于 Vue+Spring 前后端分離管理系統(tǒng)ELAdmin,大愛
想接私活時薪再翻一倍,建議根據(jù)這幾個開源的SpringBoot項目
總結(jié)
以上是生活随笔為你收集整理的干掉 if 语句,一个不留的那种!的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: SpringBoot+Vue 完整的外卖
- 下一篇: Java项目中如何更优雅的处理空值?