<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          還在寫大量 if 來判斷?試試用一個(gè)規(guī)則執(zhí)行器來替代它

          共 5807字,需瀏覽 12分鐘

           ·

          2021-06-10 02:33

          往期熱門文章:

          1、Spring Boot中的線程池,你真的會用么?

          2、重磅推薦幾個(gè)接私活的腳手架利器!

          3、MySQL究竟是怎么執(zhí)行的?看完終于不糾結(jié)了

          4、25種代碼壞味道總結(jié)+優(yōu)化示例

          5、如何優(yōu)雅處理重復(fù)請求/并發(fā)請求?
          作者 | 老鄭
          來源 | https://juejin.cn/post/6951764927958745124

          業(yè)務(wù)場景

          近日在公司領(lǐng)到一個(gè)小需求,需要對之前已有的試用用戶申請規(guī)則進(jìn)行拓展。我們的場景大概如下所示:
          if (是否海外用戶) { return false;}
          if (刷單用戶) { return false;}
          if (未付費(fèi)用戶 && 不再服務(wù)時(shí)段) { return false}
          if (轉(zhuǎn)介紹用戶 || 付費(fèi)用戶 || 內(nèi)推用戶) { return true;}
          按照上述的條件我們可以得出的結(jié)論是:
          • 咱們的的主要流程主要是基于 and 或者 or 的關(guān)系。
          • 如果有一個(gè)不匹配的話,其實(shí)咱們后續(xù)的流程是不用執(zhí)行的,就是需要具備一個(gè)短路的功能。
          • 對于目前的現(xiàn)狀來說,我如果在原有的基礎(chǔ)上來改,只要稍微注意一下解決需求不是很大的問題,但是說后面可維護(hù)性非常差。
          后面進(jìn)過權(quán)衡過后,我還是決定將這個(gè)部分進(jìn)行重構(gòu)一下。

          規(guī)則執(zhí)行器

          針對這個(gè)需求,我首先梳理了一下咱們規(guī)則執(zhí)行器大概的設(shè)計(jì), 然后我設(shè)計(jì)了一個(gè) V1 版本和大家一起分享一下,如果大家也有這樣的 case 可以給我分享留言,下面部分主要是設(shè)計(jì)和實(shí)現(xiàn)的流程和 code.

          規(guī)則執(zhí)行器的設(shè)計(jì)


          對于規(guī)則的抽象并實(shí)現(xiàn)規(guī)則

          // 業(yè)務(wù)數(shù)據(jù)@Datapublic 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ī)則- 例子1public 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ī)則- 例子2public 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)系匹配到一個(gè)就返回 true return true; } } // or 關(guān)系一個(gè)都匹配不到就返回 false return false; }}

          執(zhí)行器的調(diào)用

          public class RuleServiceTest {
          @org.junit.Test public void execute() { //規(guī)則執(zhí)行器 //優(yōu)點(diǎn):比較簡單,每個(gè)規(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):

          • 比較簡單,每個(gè)規(guī)則可以獨(dú)立,將規(guī)則,數(shù)據(jù),執(zhí)行器拆分出來,調(diào)用方比較規(guī)整;
          • 我在 Rule 模板類中定義 convert 方法做參數(shù)的轉(zhuǎn)換這樣可以能夠,為特定 rule 需要的場景數(shù)據(jù)提供拓展。

          缺點(diǎn):

          • 上下 rule 有數(shù)據(jù)依賴性,如果直接修改公共傳輸對象 dto 這樣設(shè)計(jì)不是很合理,建議提前構(gòu)建數(shù)據(jù)。

          最近熱文閱讀:

          1、重磅推薦幾個(gè)接私活的腳手架利器!

          2、MySQL究竟是怎么執(zhí)行的?看完終于不糾結(jié)了

          3、25種代碼壞味道總結(jié)+優(yōu)化示例

          4、如何優(yōu)雅處理重復(fù)請求/并發(fā)請求?

          5、使用 Redis 實(shí)現(xiàn)一個(gè)輕量級的搜索引擎

          6、比MySQL快801倍,OLAP兩大新秀ClickHouse和Doris到底怎么選?

          7、從 0 到 1 手把手教你制作酷炫可視化大屏

          8、巧用 Stream API 優(yōu)化 Java 代碼

          9、最牛逼的故障診斷工具!秒級定位線上問題

          10、一次線上 JVM 調(diào)優(yōu)實(shí)踐,F(xiàn)ullGC 40 次/天到 10 天一次的優(yōu)化過程

          關(guān)注公眾號,你想要的Java都在這里

          瀏覽 36
          點(diǎn)贊
          評論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          評論
          圖片
          表情
          推薦
          點(diǎn)贊
          評論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  最爽亂倫刺激對白1 | 激情综合自拍 | 香蕉视频黄在线观看 | 珍藏3年极品人妻疯狂3p | 痴汉无码一区二区 |