Java多條件判斷場景中規(guī)則執(zhí)行器的設(shè)計
業(yè)務(wù)場景
近日在公司領(lǐng)到一個小需求,需要對之前已有的試用用戶申請規(guī)則進(jìn)行拓展。我們的場景大概如下所示:
if (是否海外用戶) { return false; } if (刷單用戶) { return false; } if (未付費(fèi)用戶 && 不再服務(wù)時段) { return false } if (轉(zhuǎn)介紹用戶 || 付費(fèi)用戶 || 內(nèi)推用戶) { return true; }
按照上述的條件我們可以得出的結(jié)論是:
- 咱們的的主要流程主要是基于 and 或者 or 的關(guān)系。
- 如果有一個不匹配的話,其實咱們后續(xù)的流程是不用執(zhí)行的,就是需要具備一個短路的功能。
- 對于目前的現(xiàn)狀來說,我如果在原有的基礎(chǔ)上來該,只要稍微注意一下解決需求不是很大的問題,但是說后面可維護(hù)性非常差。
后面進(jìn)過權(quán)衡過后,我還是決定將這個部分進(jì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; } @Override public boolean execute(RuleDto dto) { return executeRule(convert(dto)); } protected <T> boolean executeRule(T t) { return true; } } // 具體規(guī)則- 例子1 public class AddressRule extends AbstractRule { @Override public 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 { @Override protected <T> T convert(RuleDto dto) { NationalityRuleDto nationalityRuleDto = new NationalityRuleDto(); if (dto.getAddress().startsWith(MATCH_ADDRESS_START)) { nationalityRuleDto.setNationality(MATCH_NATIONALITY_START); } return (T) nationalityRuleDto; } @Override protected <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)系匹配失敗一次,返回 false return false; } } // and 關(guān)系全部匹配成功,返回 true return true; } private boolean or(RuleDto dto, List<BaseRule> ruleList) { for (BaseRule rule : ruleList) { boolean execute = rule.execute(dto); if (execute) { // or 關(guān)系匹配到一個就返回 true return true; } } // or 關(guān)系一個都匹配不到就返回 false return false; } }
執(zhí)行器的調(diào)用
public class RuleServiceTest { @org.junit.Test public void execute() { //規(guī)則執(zhí)行器 //優(yōu)點(diǎn):比較簡單,每個規(guī)則可以獨(dú)立,將規(guī)則,數(shù)據(jù),執(zhí)行器拆分出來,調(diào)用方比較規(guī)整 //缺點(diǎn):數(shù)據(jù)依賴公共傳輸對象 dto //1. 定義規(guī)則 init rule AgeRule 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 dto RuleDto dto = new RuleDto(); dto.setAge(5); dto.setName("張三"); dto.setAddress("北京"); dto.setSubject("數(shù)學(xué)");; //3. 通過以鏈?zhǔn)秸{(diào)用構(gòu)建和執(zhí)行 rule execute boolean 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)點(diǎn)和缺點(diǎn)
優(yōu)點(diǎn):
比較簡單,每個規(guī)則可以獨(dú)立,將規(guī)則,數(shù)據(jù),執(zhí)行器拆分出來,調(diào)用方比較規(guī)整;
我在 Rule 模板類中定義 convert 方法做參數(shù)的轉(zhuǎn)換這樣可以能夠,為特定 rule 需要的場景數(shù)據(jù)提供拓展。
缺點(diǎn):上下 rule 有數(shù)據(jù)依賴性,如果直接修改公共傳輸對象 dto 這樣設(shè)計不是很合理,建議提前構(gòu)建數(shù)據(jù)。
到此這篇關(guān)于Java多條件判斷場景中規(guī)則執(zhí)行器的設(shè)計的文章就介紹到這了,更多相關(guān)Java多條件判斷規(guī)則執(zhí)行器 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
spring boot整合jsp及設(shè)置啟動頁面的方法
這篇文章主要給大家介紹了關(guān)于spring boot整合jsp及設(shè)置啟動頁面的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用spring boot具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-09-09怎樣將一個JAR包添加到Java應(yīng)用程序的Boot?Classpath中
本文文章給大家介紹如何將一個JAR包添加到Java應(yīng)用程序的Boot?Classpath中,本文通過實例代碼給大家介紹的非常詳細(xì),需要的的朋友參考下吧2023-11-11Java編程實現(xiàn)對象克?。◤?fù)制)代碼詳解
這篇文章主要介紹了Java編程實現(xiàn)對象克?。◤?fù)制)代碼詳解,涉及了克隆的原因,如何實現(xiàn)克隆,克隆的一般步驟,深克隆與淺克隆的介紹等相關(guān)內(nèi)容,具有一定借鑒價值,需要的朋友可以參考下。2017-11-11MybatisPlus3.3.0沒有MybatisPlusInterceptor類問題的解決方法
項目使用的是mybatis-plus-extension3.3.0依賴,然后在我使用分頁插件的時候,發(fā)現(xiàn)無法導(dǎo)入MybatisPlusInterceptor類所以本文給大家介紹了MybatisPlus3.3.0沒有MybatisPlusInterceptor類問題的解決方法,需要的朋友可以參考下2023-12-12Java實現(xiàn)飛機(jī)航班管理系統(tǒng)的思路詳解
這篇文章主要介紹了Java實現(xiàn)飛機(jī)航班管理系統(tǒng)的思路詳解,本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-07-07