聊城做网站公司聊城博达,部门网站的开发 意义,如何查看网站备案,大良网站建设服务前言#xff1a; 往往很多大需求都是通过增删改查堆出来的#xff0c;今天要一个需求if一下#xff0c;明天加个内容else扩展一下。日积月累需求也就越来越大#xff0c;扩展和维护的成本也就越来越高。往往大部分研发是不具备产品思维和整体业务需求导向的#xff0c;总以…前言 往往很多大需求都是通过增删改查堆出来的今天要一个需求if一下明天加个内容else扩展一下。日积月累需求也就越来越大扩展和维护的成本也就越来越高。往往大部分研发是不具备产品思维和整体业务需求导向的总以为写好代码完成功能即可。但这样的不考虑扩展性的实现很难让后续的需求都快速迭代久而久之就会被陷入恶性循环每天都有bug要改。 从上图可以看到这有点像螺丝和螺母通过一堆的链接组织出一棵结构树。而这种通过把相似对象(也可以称作是方法)组合成一组可被调用的结构树对象的设计思路叫做组合模式。
这种设计方式可以让你的服务组节点进行自由组合对外提供服务例如你有三个原子校验功能(A身份证、B银行卡、C手机号)服务并对外提供调用使用。有些调用方需要使用AB组合有些调用方需要使用到CBA组合还有一些可能只使用三者中的一个。那么这个时候你就可以使用组合模式进行构建服务对于不同类型的调用方配置不同的组织关系树而这个树结构你可以配置到数据库中也可以不断的通过图形界面来控制树结构。 用一坨坨代码实现 公司里要都是这样的程序员绝对省下不少成本根本不要搭建微服务一个工程搞定所有业务
package oom.lm.design;import org.slf4j.Logger;
import org.slf4j.LoggerFactory;public class EngineController {private Logger logger LoggerFactory.getLogger(EngineController.class);public String process(final String userId, final String userSex, final int userAge) {logger.info(ifelse实现方式判断用户结果。userId{} userSex{} userAge{}, userId, userSex, userAge);if (man.equals(userSex)) {if (userAge 25) {return 果实A;}if (userAge 25) {return 果实B;}}if (woman.equals(userSex)) {if (userAge 25) {return 果实C;}if (userAge 25) {return 果实D;}}return null;}}
除了我们说的扩展性和每次的维护以外这样的代码实现起来是最快的。而且从样子来看也很适合新人理解。 但是我劝你别写写这样代码不是被扣绩效就是被开除。
package oom.lm.design.test;import oom.lm.design.EngineController;
import org.junit.Test;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;public class ApiTest {private Logger logger LoggerFactory.getLogger(ApiTest.class);Testpublic void test_EngineController() {EngineController engineController new EngineController();String process engineController.process(Oli09pLkdjh, man, 29);logger.info(测试结果{}, process);}}
组合模式重构代码 组合模式模型结构 树节点逻辑过滤器接口
package oom.lm.design.domain.service.logic;import oom.lm.design.domain.model.vo.TreeNodeLink;import java.util.List;
import java.util.Map;/******/
public interface LogicFilter {/*** 逻辑决策器** param matterValue 决策值* param treeNodeLineInfoList 决策节点* return 下一个节点Id*/Long filter(String matterValue, ListTreeNodeLink treeNodeLineInfoList);/*** 获取决策值** param decisionMatter 决策物料* return 决策值*/String matterValue(Long treeId, String userId, MapString, String decisionMatter);}
决策抽象类提供基础服务
package oom.lm.design.domain.service.logic;import oom.lm.design.domain.model.vo.TreeNodeLink;import java.util.List;
import java.util.Map;/******/
public abstract class BaseLogic implements LogicFilter {Overridepublic Long filter(String matterValue, ListTreeNodeLink treeNodeLinkList) {for (TreeNodeLink nodeLine : treeNodeLinkList) {if (decisionLogic(matterValue, nodeLine)){return nodeLine.getNodeIdTo();}}return 0L;}Overridepublic abstract String matterValue(Long treeId, String userId, MapString, String decisionMatter);private boolean decisionLogic(String matterValue, TreeNodeLink nodeLink) {switch (nodeLink.getRuleLimitType()) {case 1:return matterValue.equals(nodeLink.getRuleLimitValue());case 2:return Double.parseDouble(matterValue) Double.parseDouble(nodeLink.getRuleLimitValue());case 3:return Double.parseDouble(matterValue) Double.parseDouble(nodeLink.getRuleLimitValue());case 4:return Double.parseDouble(matterValue) Double.parseDouble(nodeLink.getRuleLimitValue());case 5:return Double.parseDouble(matterValue) Double.parseDouble(nodeLink.getRuleLimitValue());default:return false;}}}
树节点逻辑实现类
package oom.lm.design.domain.service.logic.impl;import oom.lm.design.domain.service.logic.BaseLogic;import java.util.Map;public class UserAgeFilter extends BaseLogic {Overridepublic String matterValue(Long treeId, String userId, MapString, String decisionMatter) {return decisionMatter.get(age);}}
package oom.lm.design.domain.service.logic.impl;import oom.lm.design.domain.service.logic.BaseLogic;import java.util.Map;public class UserGenderFilter extends BaseLogic {Overridepublic String matterValue(Long treeId, String userId, MapString, String decisionMatter) {return decisionMatter.get(gender);}}
以上两个决策逻辑的节点获取值的方式都非常简单只是获取用户的入参即可。实际的业务开发可以从数据库、RPC接口、缓存运算等各种方式获取。 决策引擎接口定义
package oom.lm.design.domain.service.engine;import oom.lm.design.domain.model.aggregates.TreeRich;
import oom.lm.design.domain.model.vo.EngineResult;import java.util.Map;public interface IEngine {EngineResult process(final Long treeId, final String userId, TreeRich treeRich, final MapString, String decisionMatter);}
决策节点配置
package oom.lm.design.domain.service.engine;import oom.lm.design.domain.service.logic.LogicFilter;
import oom.lm.design.domain.service.logic.impl.UserAgeFilter;
import oom.lm.design.domain.service.logic.impl.UserGenderFilter;import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;public class EngineConfig {static MapString, LogicFilter logicFilterMap;static {logicFilterMap new ConcurrentHashMap();logicFilterMap.put(userAge, new UserAgeFilter());logicFilterMap.put(userGender, new UserGenderFilter());}public MapString, LogicFilter getLogicFilterMap() {return logicFilterMap;}public void setLogicFilterMap(MapString, LogicFilter logicFilterMap) {this.logicFilterMap logicFilterMap;}}
基础决策引擎功能
package oom.lm.design.domain.service.engine;import oom.lm.design.domain.model.aggregates.TreeRich;
import oom.lm.design.domain.model.vo.EngineResult;
import oom.lm.design.domain.model.vo.TreeNode;
import oom.lm.design.domain.model.vo.TreeRoot;
import oom.lm.design.domain.service.logic.LogicFilter;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;import java.util.Map;
public abstract class EngineBase extends EngineConfig implements IEngine {private Logger logger LoggerFactory.getLogger(EngineBase.class);Overridepublic abstract EngineResult process(Long treeId, String userId, TreeRich treeRich, MapString, String decisionMatter);protected TreeNode engineDecisionMaker(TreeRich treeRich, Long treeId, String userId, MapString, String decisionMatter) {TreeRoot treeRoot treeRich.getTreeRoot();MapLong, TreeNode treeNodeMap treeRich.getTreeNodeMap();// 规则树根IDLong rootNodeId treeRoot.getTreeRootNodeId();TreeNode treeNodeInfo treeNodeMap.get(rootNodeId);//节点类型[NodeType]1子叶、2果实while (treeNodeInfo.getNodeType().equals(1)) {String ruleKey treeNodeInfo.getRuleKey();LogicFilter logicFilter logicFilterMap.get(ruleKey);String matterValue logicFilter.matterValue(treeId, userId, decisionMatter);Long nextNode logicFilter.filter(matterValue, treeNodeInfo.getTreeNodeLinkList());treeNodeInfo treeNodeMap.get(nextNode);logger.info(决策树引擎{} userId{} treeId{} treeNode{} ruleKey{} matterValue{}, treeRoot.getTreeName(), userId, treeId, treeNodeInfo.getTreeNodeId(), ruleKey, matterValue);}return treeNodeInfo;}}
这里主要提供决策树流程的处理过程有点像通过链路的关系(性别、年龄)在二叉树中寻找果实节点的过程。 决策引擎的实现
package oom.lm.design.domain.service.engine.impl;import oom.lm.design.domain.model.aggregates.TreeRich;
import oom.lm.design.domain.model.vo.EngineResult;
import oom.lm.design.domain.model.vo.TreeNode;
import oom.lm.design.domain.service.engine.EngineBase;import java.util.Map;public class TreeEngineHandle extends EngineBase {Overridepublic EngineResult process(Long treeId, String userId, TreeRich treeRich, MapString, String decisionMatter) {// 决策流程TreeNode treeNode engineDecisionMaker(treeRich, treeId, userId, decisionMatter);// 决策结果return new EngineResult(userId, treeId, treeNode.getTreeNodeId(), treeNode.getNodeValue());}}
测试验证
package oom.lm.design.test;import com.alibaba.fastjson.JSON;
import oom.lm.design.domain.model.aggregates.TreeRich;
import oom.lm.design.domain.model.vo.EngineResult;
import oom.lm.design.domain.model.vo.TreeNode;
import oom.lm.design.domain.model.vo.TreeNodeLink;
import oom.lm.design.domain.model.vo.TreeRoot;
import oom.lm.design.domain.service.engine.IEngine;
import oom.lm.design.domain.service.engine.impl.TreeEngineHandle;
import org.junit.Before;
import org.junit.Test;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;public class ApiTest {private Logger logger LoggerFactory.getLogger(ApiTest.class);private TreeRich treeRich;Beforepublic void init() {// 节点1TreeNode treeNode_01 new TreeNode();treeNode_01.setTreeId(10001L);treeNode_01.setTreeNodeId(1L);treeNode_01.setNodeType(1);treeNode_01.setNodeValue(null);treeNode_01.setRuleKey(userGender);treeNode_01.setRuleDesc(用户性别[男/女]);// 链接1-11TreeNodeLink treeNodeLink_11 new TreeNodeLink();treeNodeLink_11.setNodeIdFrom(1L);treeNodeLink_11.setNodeIdTo(11L);treeNodeLink_11.setRuleLimitType(1);treeNodeLink_11.setRuleLimitValue(man);// 链接1-12TreeNodeLink treeNodeLink_12 new TreeNodeLink();treeNodeLink_12.setNodeIdFrom(1L);treeNodeLink_12.setNodeIdTo(12L);treeNodeLink_12.setRuleLimitType(1);treeNodeLink_12.setRuleLimitValue(woman);ListTreeNodeLink treeNodeLinkList_1 new ArrayList();treeNodeLinkList_1.add(treeNodeLink_11);treeNodeLinkList_1.add(treeNodeLink_12);treeNode_01.setTreeNodeLinkList(treeNodeLinkList_1);// 节点11TreeNode treeNode_11 new TreeNode();treeNode_11.setTreeId(10001L);treeNode_11.setTreeNodeId(11L);treeNode_11.setNodeType(1);treeNode_11.setNodeValue(null);treeNode_11.setRuleKey(userAge);treeNode_11.setRuleDesc(用户年龄);// 链接11-111TreeNodeLink treeNodeLink_111 new TreeNodeLink();treeNodeLink_111.setNodeIdFrom(11L);treeNodeLink_111.setNodeIdTo(111L);treeNodeLink_111.setRuleLimitType(3);treeNodeLink_111.setRuleLimitValue(25);// 链接11-112TreeNodeLink treeNodeLink_112 new TreeNodeLink();treeNodeLink_112.setNodeIdFrom(11L);treeNodeLink_112.setNodeIdTo(112L);treeNodeLink_112.setRuleLimitType(4);treeNodeLink_112.setRuleLimitValue(25);ListTreeNodeLink treeNodeLinkList_11 new ArrayList();treeNodeLinkList_11.add(treeNodeLink_111);treeNodeLinkList_11.add(treeNodeLink_112);treeNode_11.setTreeNodeLinkList(treeNodeLinkList_11);// 节点12TreeNode treeNode_12 new TreeNode();treeNode_12.setTreeId(10001L);treeNode_12.setTreeNodeId(12L);treeNode_12.setNodeType(1);treeNode_12.setNodeValue(null);treeNode_12.setRuleKey(userAge);treeNode_12.setRuleDesc(用户年龄);// 链接12-121TreeNodeLink treeNodeLink_121 new TreeNodeLink();treeNodeLink_121.setNodeIdFrom(12L);treeNodeLink_121.setNodeIdTo(121L);treeNodeLink_121.setRuleLimitType(3);treeNodeLink_121.setRuleLimitValue(25);// 链接12-122TreeNodeLink treeNodeLink_122 new TreeNodeLink();treeNodeLink_122.setNodeIdFrom(12L);treeNodeLink_122.setNodeIdTo(122L);treeNodeLink_122.setRuleLimitType(4);treeNodeLink_122.setRuleLimitValue(25);ListTreeNodeLink treeNodeLinkList_12 new ArrayList();treeNodeLinkList_12.add(treeNodeLink_121);treeNodeLinkList_12.add(treeNodeLink_122);treeNode_12.setTreeNodeLinkList(treeNodeLinkList_12);// 节点111TreeNode treeNode_111 new TreeNode();treeNode_111.setTreeId(10001L);treeNode_111.setTreeNodeId(111L);treeNode_111.setNodeType(2);treeNode_111.setNodeValue(果实A);// 节点112TreeNode treeNode_112 new TreeNode();treeNode_112.setTreeId(10001L);treeNode_112.setTreeNodeId(112L);treeNode_112.setNodeType(2);treeNode_112.setNodeValue(果实B);// 节点121TreeNode treeNode_121 new TreeNode();treeNode_121.setTreeId(10001L);treeNode_121.setTreeNodeId(121L);treeNode_121.setNodeType(2);treeNode_121.setNodeValue(果实C);// 节点122TreeNode treeNode_122 new TreeNode();treeNode_122.setTreeId(10001L);treeNode_122.setTreeNodeId(122L);treeNode_122.setNodeType(2);treeNode_122.setNodeValue(果实D);// 树根TreeRoot treeRoot new TreeRoot();treeRoot.setTreeId(10001L);treeRoot.setTreeRootNodeId(1L);treeRoot.setTreeName(规则决策树);MapLong, TreeNode treeNodeMap new HashMap();treeNodeMap.put(1L, treeNode_01);treeNodeMap.put(11L, treeNode_11);treeNodeMap.put(12L, treeNode_12);treeNodeMap.put(111L, treeNode_111);treeNodeMap.put(112L, treeNode_112);treeNodeMap.put(121L, treeNode_121);treeNodeMap.put(122L, treeNode_122);treeRich new TreeRich(treeRoot, treeNodeMap);}Testpublic void test_tree() {logger.info(决策树组合结构信息\r\n JSON.toJSONString(treeRich));IEngine treeEngineHandle new TreeEngineHandle();MapString, String decisionMatter new HashMap();decisionMatter.put(gender, man);decisionMatter.put(age, 29);EngineResult result treeEngineHandle.process(10001L, Oli09pLkdjh, treeRich, decisionMatter);logger.info(测试结果{}, JSON.toJSONString(result));}
}
总结 1从以上的决策树场景来看组合模式主要解决的是一系列简单逻辑节点或者扩展的复杂逻辑节点在不同结构的组织下对于外部的调用是仍然可以非常简单的。 2这部分设计模式保证了开闭原则无需更改模型结构你就可以提供新的逻辑节点的使用并配合组织出新的关系树。但如果是一些功能差异化非常大的接口进行包装就会变得比较困难但也不是不能很好的处理只不过需要做一些适配和特定化的开发。
好了 至此 设计模式之组合模式(营销差异化人群发券决策树引擎搭建场景) 学习结束了 友友们 点点关注不迷路 老铁们 文章转载自: http://www.morning.jbpdk.cn.gov.cn.jbpdk.cn http://www.morning.nspzy.cn.gov.cn.nspzy.cn http://www.morning.zlnyk.cn.gov.cn.zlnyk.cn http://www.morning.lsgjf.cn.gov.cn.lsgjf.cn http://www.morning.frtb.cn.gov.cn.frtb.cn http://www.morning.hdtcj.cn.gov.cn.hdtcj.cn http://www.morning.tzpqc.cn.gov.cn.tzpqc.cn http://www.morning.sgbsr.cn.gov.cn.sgbsr.cn http://www.morning.mnclk.cn.gov.cn.mnclk.cn http://www.morning.rqlf.cn.gov.cn.rqlf.cn http://www.morning.qlhwy.cn.gov.cn.qlhwy.cn http://www.morning.xjtnp.cn.gov.cn.xjtnp.cn http://www.morning.lizpw.com.gov.cn.lizpw.com http://www.morning.bzpwh.cn.gov.cn.bzpwh.cn http://www.morning.hhrpy.cn.gov.cn.hhrpy.cn http://www.morning.hslgq.cn.gov.cn.hslgq.cn http://www.morning.gxklx.cn.gov.cn.gxklx.cn http://www.morning.oumong.com.gov.cn.oumong.com http://www.morning.lmmyl.cn.gov.cn.lmmyl.cn http://www.morning.yqtry.cn.gov.cn.yqtry.cn http://www.morning.wjlrw.cn.gov.cn.wjlrw.cn http://www.morning.cbmqq.cn.gov.cn.cbmqq.cn http://www.morning.crqbt.cn.gov.cn.crqbt.cn http://www.morning.tphrx.cn.gov.cn.tphrx.cn http://www.morning.mrlkr.cn.gov.cn.mrlkr.cn http://www.morning.gsjw.cn.gov.cn.gsjw.cn http://www.morning.rqknq.cn.gov.cn.rqknq.cn http://www.morning.qsmdd.cn.gov.cn.qsmdd.cn http://www.morning.jkpnm.cn.gov.cn.jkpnm.cn http://www.morning.lrylj.cn.gov.cn.lrylj.cn http://www.morning.jjmrx.cn.gov.cn.jjmrx.cn http://www.morning.hysqx.cn.gov.cn.hysqx.cn http://www.morning.wtrjq.cn.gov.cn.wtrjq.cn http://www.morning.rntgy.cn.gov.cn.rntgy.cn http://www.morning.prmbn.cn.gov.cn.prmbn.cn http://www.morning.nqwkn.cn.gov.cn.nqwkn.cn http://www.morning.qgfkn.cn.gov.cn.qgfkn.cn http://www.morning.xplng.cn.gov.cn.xplng.cn http://www.morning.nqlcj.cn.gov.cn.nqlcj.cn http://www.morning.rlfr.cn.gov.cn.rlfr.cn http://www.morning.pbzlh.cn.gov.cn.pbzlh.cn http://www.morning.lywpd.cn.gov.cn.lywpd.cn http://www.morning.jjhrj.cn.gov.cn.jjhrj.cn http://www.morning.ryywf.cn.gov.cn.ryywf.cn http://www.morning.pdwzr.cn.gov.cn.pdwzr.cn http://www.morning.fdrwk.cn.gov.cn.fdrwk.cn http://www.morning.tymnr.cn.gov.cn.tymnr.cn http://www.morning.mwcqz.cn.gov.cn.mwcqz.cn http://www.morning.xjbtb.cn.gov.cn.xjbtb.cn http://www.morning.grlth.cn.gov.cn.grlth.cn http://www.morning.xbxks.cn.gov.cn.xbxks.cn http://www.morning.npcxk.cn.gov.cn.npcxk.cn http://www.morning.gwkwt.cn.gov.cn.gwkwt.cn http://www.morning.jhwwr.cn.gov.cn.jhwwr.cn http://www.morning.fqyxb.cn.gov.cn.fqyxb.cn http://www.morning.jfqqs.cn.gov.cn.jfqqs.cn http://www.morning.kkqgf.cn.gov.cn.kkqgf.cn http://www.morning.bwttj.cn.gov.cn.bwttj.cn http://www.morning.zwgrf.cn.gov.cn.zwgrf.cn http://www.morning.ssjee.cn.gov.cn.ssjee.cn http://www.morning.qrgfw.cn.gov.cn.qrgfw.cn http://www.morning.fdsbs.cn.gov.cn.fdsbs.cn http://www.morning.wplbs.cn.gov.cn.wplbs.cn http://www.morning.wnnfh.cn.gov.cn.wnnfh.cn http://www.morning.ylpl.cn.gov.cn.ylpl.cn http://www.morning.wknj.cn.gov.cn.wknj.cn http://www.morning.qhkx.cn.gov.cn.qhkx.cn http://www.morning.fbdkb.cn.gov.cn.fbdkb.cn http://www.morning.mbmtn.cn.gov.cn.mbmtn.cn http://www.morning.pcqdf.cn.gov.cn.pcqdf.cn http://www.morning.qynpw.cn.gov.cn.qynpw.cn http://www.morning.hqwtm.cn.gov.cn.hqwtm.cn http://www.morning.hfxks.cn.gov.cn.hfxks.cn http://www.morning.ktnmg.cn.gov.cn.ktnmg.cn http://www.morning.mrfjr.cn.gov.cn.mrfjr.cn http://www.morning.mswkd.cn.gov.cn.mswkd.cn http://www.morning.nd-test.com.gov.cn.nd-test.com http://www.morning.bsxws.cn.gov.cn.bsxws.cn http://www.morning.azxey.cn.gov.cn.azxey.cn http://www.morning.xbhpm.cn.gov.cn.xbhpm.cn