自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

DDD 領(lǐng)域驅(qū)動決策規(guī)則樹服務(wù)架構(gòu)設(shè)計

開發(fā) 架構(gòu)
一些大型架構(gòu)設(shè)計往往不是換一個設(shè)計模型就能徹底提升效率,還是需要人員整體素質(zhì),這是一個不斷培養(yǎng)的過程。領(lǐng)域驅(qū)動設(shè)計的思想并不只是教會程序猿寫代碼,也是非程序員以外的所有互聯(lián)網(wǎng)人員都適合學(xué)習(xí)的內(nèi)容。

前言介紹

在上一章節(jié)介紹了領(lǐng)域驅(qū)動設(shè)計的基本概念以及按照領(lǐng)域驅(qū)動設(shè)計的思想進行代碼分層,但是僅僅只是從一個簡單的分層結(jié)構(gòu)上依然沒法理解DDD以及如何去開發(fā)這樣的微服務(wù)。另外往往按照這樣分層后依然感覺和MVC也沒有什么差別,也沒有感受到帶來什么非常大的好處。那么問題出在哪呢?我個人覺得DDD學(xué)起來更像是一套指導(dǎo)思想,不斷的將學(xué)習(xí)者引入到領(lǐng)域觸發(fā)的思維中去,而這恰恰也是最難學(xué)習(xí)的地方。時而感覺會了,而實際開發(fā)中又不對,本來已經(jīng)拆解清晰了,但怎么又那么像MVC了。甚至懷疑自己,我在干嘛?

無論是DDD、MVC,他們更像是家里三居或者四局的格局,每一種格局方式都是為了更好的實現(xiàn)對應(yīng)架構(gòu)下的設(shè)計思想。但,不是說給你一個通用的架構(gòu)模式,你就能開發(fā)出干凈(高內(nèi)聚)、整潔(低耦合)、漂亮(模塊化)的代碼。這就像是你家住三居、他家也住三居,但是你們屋子的舒適情況就一樣嗎?{再有,你家里會把廁所安在廚房嗎?但你的代碼是否這么干過,不合理的擺放導(dǎo)致重構(gòu)延期。}

另外DDD之所以看著簡單但又不那么好落地,個人認(rèn)為很重要就是領(lǐng)域思想,DDD只是指導(dǎo)但是不能把互聯(lián)網(wǎng)天下每一個業(yè)務(wù)行為開發(fā)都拿出來舉例子給你看,每個領(lǐng)域需要設(shè)計。所以需要一些領(lǐng)域?qū)<遥a(chǎn)品+架構(gòu)+不是杠精的程序猿}來討論梳理,將業(yè)務(wù)形態(tài)設(shè)計出合理的架構(gòu)&代碼。

案例目標(biāo)

本案例通過一個商品下單規(guī)則的場景來進行演示DDD;

  1. 假設(shè)產(chǎn)品需求業(yè)務(wù)運行人員可以對不同的商品配置一些規(guī)則,這些規(guī)則可以滿足不同用戶類型可以下單不同商品。
  2. 另外一些行為規(guī)則是會隨著業(yè)務(wù)發(fā)展而增加或者變動的,所以不能寫死{寫死太嚇人了}。
  3. 數(shù)據(jù)庫的PO類不應(yīng)該被外部服務(wù)調(diào)用,這也是必須的。如果你開發(fā)過很多系統(tǒng),那么可能已經(jīng)吃過虧并意識到這個問題。
  4. 按照DDD思想我們嘗試需要設(shè)計一個規(guī)則引擎的服務(wù),通過給外部提供非常簡單的接口(application)來獲取最終結(jié)果。
  5. 通過這樣的案例可以很容易的感受到目前的四層架構(gòu)確實在實現(xiàn)DDD思想上有很多的幫助。

如圖;DDD分層結(jié)構(gòu) | 指導(dǎo)設(shè)計架構(gòu)

bugstack蟲洞棧 & 分層結(jié)構(gòu)

DDD思想 · 開發(fā)設(shè)計

通過領(lǐng)域驅(qū)動設(shè)計的思想,從領(lǐng)域知識中提取和劃分為一個一個的子領(lǐng)域(核心子域,通用子域,支撐子域),并在子領(lǐng)域上建立模型。那么在技術(shù)實現(xiàn)上就需要去支撐這種建模,以使我們的代碼模塊獨立、免污染、易于擴展。

在上面我們提到需要開發(fā)一個可擴展使用的規(guī)則樹,那么如果只是單純的一次性需求,最快的方式是if語句就搞定了。但是為了使這個領(lǐng)域服務(wù)具備良好的使用和擴展性,我們需要做些拆分,那么如下;

  1. 你是否想過系統(tǒng)在過濾過則的時候其實就像執(zhí)行一棵二叉樹一樣非左即右側(cè),每一條線上都有著執(zhí)行條件,通過判斷來達到最終的結(jié)果。
  2. 按照樹形結(jié)構(gòu)我們將定義出來四個類;樹、節(jié)點、果實、指向線(From-To),用于描述我們的規(guī)則行為。
  3. 再此基礎(chǔ)上需要實現(xiàn)一個邏輯定義與規(guī)則樹執(zhí)行引擎,通過統(tǒng)一的引擎服務(wù)來執(zhí)行我們每次配置好的規(guī)則樹。

如圖;領(lǐng)域開發(fā)設(shè)計服務(wù)

bugstack蟲洞棧 & 領(lǐng)域開發(fā)設(shè)計服務(wù)

工程模型

itstack-demo-ddd-02
└── src
├── main
│ ├── java
│ │ └── org.itstack.demo
│ │ ├── application
│ │ │ ├── MallRuleService.java
│ │ │ └── MallTreeService.java
│ │ ├── domain
│ │ │ ├── rule
│ │ │ │ ├── model
│ │ │ │ │ ├── aggregates
│ │ │ │ │ │ └── UserRichInfo.java
│ │ │ │ │ └── vo
│ │ │ │ │ ├── DecisionMatter.java
│ │ │ │ │ ├── EngineResult.java
│ │ │ │ │ ├── TreeNodeInfo.java
│ │ │ │ │ ├── TreeNodeLineInfo.java
│ │ │ │ │ └── UserSchool.java
│ │ │ │ ├── repository
│ │ │ │ │ └── IRuleRepository.java
│ │ │ │ └── service
│ │ │ │ ├── engine
│ │ │ │ │ ├── impl
│ │ │ │ │ └── EngineFilter.java
│ │ │ │ ├── logic
│ │ │ │ │ ├── impl
│ │ │ │ │ └── LogicFilter.java
│ │ │ │ └── MallRuleServiceImpl.java
│ │ │ └── tree
│ │ │ ├── model
│ │ │ │ ├── aggregates
│ │ │ │ │ └── TreeCollect.java
│ │ │ │ └── vo
│ │ │ │ ├── TreeInfo.java
│ │ │ │ └── TreeRulePoint.java
│ │ │ ├── repository
│ │ │ │ └── ITreeRepository.java
│ │ │ └── service
│ │ │ └── MallTreeServiceImpl.java
│ │ ├── infrastructure
│ │ │ ├── common
│ │ │ │ └── Constants.java
│ │ │ ├── dao
│ │ │ │ ├── RuleTreeDao.java
│ │ │ │ ├── RuleTreeNodeDao.java
│ │ │ │ └── RuleTreeNodeLineDao.java
│ │ │ ├── po
│ │ │ │ ├── RuleTree.java
│ │ │ │ ├── RuleTreeConfig.java
│ │ │ │ ├── RuleTreeNode.java
│ │ │ │ └── RuleTreeNodeLine.java
│ │ │ ├── repository
│ │ │ │ ├── cache
│ │ │ │ │ └── RuleCacheRepository.java
│ │ │ │ ├── mysql
│ │ │ │ │ ├── RuleMysqlRepository.java
│ │ │ │ │ └── TreeMysqlRepository.java
│ │ │ │ ├── RuleRepository.java
│ │ │ │ └── TreeRepository.java
│ │ │ └── util
│ │ │ └── CacheUtil.java
│ │ ├── interfaces
│ │ │ ├── dto
│ │ │ │ ├── DecisionMatterDTO.java
│ │ │ │ └── TreeDTO.java
│ │ │ └── DDDController.java
│ │ └── DDDApplication.java
│ └── resources
│ ├── mybatis
│ └── application.yml
└── test
└── java
└── org.itstack.demo.test
└── ApiTest.java

application應(yīng)用層

application/MallRuleService.java | 應(yīng)用層定義接口服務(wù),也可以適當(dāng)做簡單包裝

/**
* 商超規(guī)則過濾服務(wù);提供規(guī)則樹決策功能
* 微信公眾號:bugstack蟲洞棧 | 專注原創(chuàng)技術(shù)專題案例
* 論壇:http://bugstack.cn
* Create by 付政委 on @2019
*/
public interface MallRuleService {

/**
* 決策服務(wù)
* @param matter 決策物料
* @return
EngineResult process(final DecisionMatter matter);

}

domain領(lǐng)域?qū)?/h2>

domain中有兩個領(lǐng)域服務(wù);規(guī)則樹信息領(lǐng)域、規(guī)則執(zhí)行領(lǐng)域,通過合理的抽象化來實現(xiàn)高內(nèi)聚、低耦合的模塊化服務(wù)

domain/service/MallRuleServiceImpl.java | 領(lǐng)域?qū)又械膕ervice來實現(xiàn)應(yīng)用層接口

/**
* 規(guī)則樹服務(wù);提供規(guī)則規(guī)律功能
*
* 1、rule包下只進行規(guī)則決策領(lǐng)域的處理
* 2、封裝決策行為到領(lǐng)域模型中,外部只需要調(diào)用和處理結(jié)果即可
* 3、可以擴展不同的決策引擎進行統(tǒng)一管理
*
* 微信公眾號:bugstack蟲洞棧 | 專注原創(chuàng)技術(shù)專題案例
* 論壇:http://bugstack.cn
* Create by 付政委 on @2019
*/
@Service("mallRuleService")
public class MallRuleServiceImpl implements MallRuleService {

private Logger logger = LoggerFactory.getLogger(MallRuleServiceImpl.class);

@Resource(name = "ruleEngineHandle")
private EngineFilter ruleEngineHandle;

@Override
public EngineResult process(DecisionMatter matter){
try {
return ruleEngineHandle.process(matter);
} catch (Exception e) {
logger.error("決策引擎執(zhí)行失敗", e);
return new EngineResult(false);
}
}

}

domain/service/logic/LogicFilter.java | 邏輯決策定義

/**
* 微信公眾號:bugstack蟲洞棧 | 專注原創(chuàng)技術(shù)專題案例
* 論壇:http://bugstack.cn
* Create by 付政委 on @2019
*/
public interface LogicFilter {

/**
* 邏輯決策器
* @param matterValue 決策值
* @param treeNodeLineInfoList 決策節(jié)點
* @return
Long filter(String matterValue, List<TreeNodeLineInfo> treeNodeLineInfoList);

/**
* 獲取決策值
*
* @param decisionMatter 決策物料
* @return
String matterValue(DecisionMatter decisionMatter);

}

domain/service/engine/EngineFilter.java | 引擎執(zhí)行定義

/**
* 微信公眾號:bugstack蟲洞棧 | 專注原創(chuàng)技術(shù)專題案例
* 論壇:http://bugstack.cn
* Create by 付政委 on @2019
*/
public interface EngineFilter {

EngineResult process(final DecisionMatter matter) throws Exception;

}

infrastructure基礎(chǔ)層

1、實現(xiàn)領(lǐng)域?qū)觽}儲定義 2、數(shù)據(jù)庫操作為非業(yè)務(wù)屬性的功能操作 3、在倉儲實現(xiàn)層進行組合裝配DAO&Redis&Cache等

infrastructure/repository/RuleRepository.java

/**
* 微信公眾號:bugstack蟲洞棧 | 專注原創(chuàng)技術(shù)專題案例
* 論壇:http://bugstack.cn
* Create by 付政委 on @2019
*/
public interface EngineFilter {

EngineResult process(final DecisionMatter matter) throws Exception;

}

interfaces接口層

1、包裝應(yīng)用接口對外提供api 2、外部傳輸對象采用DTO類,主要為了避免內(nèi)部類被污染{不斷的迭代的需求會在類中增加很多字段} 3、目前依然是提供的Http服務(wù),如果提供的rpc服務(wù),將需要對外提供可引用jar

interfaces/DDDController.java

**
* 微信公眾號:bugstack蟲洞棧 | 歡迎關(guān)注學(xué)習(xí)專題案例
* 論壇:http://bugstack.cn
* Create by 付政委 on @2019
*/
@Controller
public class DDDController {

private Logger logger = LoggerFactory.getLogger(DDDController.class);

@Resource
private MallTreeService mallTreeService;
@Resource
private MallRuleService mallRuleService;

/**
* 測試接口:http://localhost:8080/api/tree/queryTreeSummaryInfo
* 請求參數(shù):{"treeId":10001}
*/
@RequestMapping(path = "/api/tree/queryTreeSummaryInfo", method = RequestMethod.POST)
@ResponseBody
public ResponseEntity queryTreeSummaryInfo(@RequestBody TreeDTO request) {
String reqStr = JSON.toJSONString(request);
try {
logger.info("查詢規(guī)則樹信息{}Begin req:{}", request.getTreeId(), reqStr);
TreeCollect treeCollect = mallTreeService.queryTreeSummaryInfo(request.getTreeId());
logger.info("查詢規(guī)則樹信息{}End res:{}", request.getTreeId(), JSON.toJSON(treeCollect));
return new ResponseEntity<>(treeCollect, HttpStatus.OK);
} catch (Exception e) {
logger.error("查詢規(guī)則樹信息{}Error req:{}", request.getTreeId(), reqStr, e);
return new ResponseEntity<>(e.getMessage(), HttpStatus.OK);
}
}

/**
* 測試接口:http://localhost:8080/api/tree/decisionRuleTree
* 請求參數(shù):{"treeId":10001,"userId":"fuzhengwei","valMap":{"gender":"man","age":"25"}}
*/
@RequestMapping(path = "/api/tree/decisionRuleTree", method = RequestMethod.POST)
@ResponseBody
public ResponseEntity decisionRuleTree(@RequestBody DecisionMatterDTO request) {
String reqStr = JSON.toJSONString(request);
try {
logger.info("規(guī)則樹行為信息決策{}Begin req:{}", request.getTreeId(), reqStr);
DecisionMatter decisionMatter = new DecisionMatter();
decisionMatter.setTreeId(request.getTreeId());
decisionMatter.setUserId(request.getUserId());
decisionMatter.setValMap(request.getValMap());
EngineResult engineResult = mallRuleService.process(decisionMatter);
logger.info("規(guī)則樹行為信息決策{}End res:{}", request.getTreeId(), JSON.toJSON(engineResult));
return new ResponseEntity<>(engineResult, HttpStatus.OK);
} catch (Exception e) {
logger.error("規(guī)則樹行為信息決策{}Error req:{}", request.getTreeId(), reqStr, e);
return new ResponseEntity<>(e.getMessage(), HttpStatus.OK);
}
}

}

測試驗證

規(guī)則樹結(jié)構(gòu){數(shù)據(jù)庫轉(zhuǎn)Json} | 可自行定義

{
"treeNodeMap": {
"1": {
"nodeType": 1,
"ruleDesc": "用戶性別[男/女]",
"ruleKey": "userGender",
"treeId": 10001,
"treeNodeId": 1,
"treeNodeLineInfoList": [
{
"nodeIdFrom": 1,
"nodeIdTo": 11,
"ruleLimitType": 1,
"ruleLimitValue": "man"
},
{
"nodeIdFrom": 1,
"nodeIdTo": 12,
"ruleLimitType": 1,
"ruleLimitValue": "woman"
}
]
},
"11": {
"nodeType": 1,
"ruleDesc": "用戶年齡",
"ruleKey": "userAge",
"treeId": 10001,
"treeNodeId": 11,
"treeNodeLineInfoList": [
{
"nodeIdFrom": 11,
"nodeIdTo": 111,
"ruleLimitType": 3,
"ruleLimitValue": "25"
},
{
"nodeIdFrom": 11,
"nodeIdTo": 112,
"ruleLimitType": 3,
"ruleLimitValue": "25"
}
]
},
"12": {
"nodeType": 1,
"ruleDesc": "用戶年齡",
"ruleKey": "userAge",
"treeId": 10001,
"treeNodeId": 12,
"treeNodeLineInfoList": [
{
"nodeIdFrom": 12,
"nodeIdTo": 121,
"ruleLimitType": 3,
"ruleLimitValue": "25"
},
{
"nodeIdFrom": 12,
"nodeIdTo": 122,
"ruleLimitType": 3,
"ruleLimitValue": "25"
}
]
},
"111": {
"nodeType": 2,
"nodeValue": "果實A",
"treeId": 10001,
"treeNodeId": 111,
"treeNodeLineInfoList": [ ]
},
"112": {
"nodeType": 2,
"nodeValue": "果實B",
"treeId": 10001,
"treeNodeId": 112,
"treeNodeLineInfoList": [ ]
},
"121": {
"nodeType": 2,
"nodeValue": "果實C",
"treeId": 10001,
"treeNodeId": 121,
"treeNodeLineInfoList": [ ]
},
"122": {
"nodeType": 2,
"nodeValue": "果實D",
"treeId": 10001,
"treeNodeId": 122,
"treeNodeLineInfoList": [ ]
}
},
"treeRoot": {
"treeId": 10001,
"treeName": "購物分類規(guī)則樹",
"treeRootNodeId": 1
}
}

通過postman調(diào)用 | raw => json

查詢規(guī)則樹信息 測試接口:http://localhost:8080/api/tree/decisionRuleTree 請求參數(shù):{"treeId":10001}

{
"treeInfo": {
"treeId": 10001,
"treeName": "購物分類規(guī)則樹",
"treeDesc": "用于分類不同類型用戶可購物范圍",
"nodeCount": 7,
"lineCount": 6
},
"treeRulePointList": [
{
"ruleKey": "userGender",
"ruleDesc": "用戶性別[男/女]"
},
{
"ruleKey": "userAge",
"ruleDesc": "用戶年齡"
}
]
}

微信公眾號:bugstack蟲洞棧 & 查詢規(guī)則樹信息

規(guī)則樹行為信息決策 測試接口:http://localhost:8080/api/tree/decisionRuleTree 請求參數(shù):{"treeId":10001}

{
"userId": "fuzhengwei",
"treeId": 10001,
"nodeId": 112,
"nodeValue": "果實B",
"success": true
}

微信公眾號:bugstack蟲洞棧 & 規(guī)則樹行為信息決策

  .   ____          _            __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
\\/ ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_\__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v2.0.5.RELEASE)

2019-10-19 18:22:05.672 INFO 13820 --- [ main] org.itstack.demo.DDDApplication : Starting DDDApplication on fuzhengwei-PC with PID 13820 (E:\itstack\itstack.org\itstack-demo-ddd-02\target\classes started by fuzhengwei in E:\itstack\itstack.org\itstack-demo-ddd-02)
2019-10-19 18:22:05.675 INFO 13820 --- [ main] org.itstack.demo.DDDApplication : No active profile set, falling back to default profiles: default
2019-10-19 18:22:05.952 INFO 13820 --- [ main] ConfigServletWebServerApplicationContext : Refreshing org.springframework.boot.web.servlet.context.AnnotationConfigServletWebServerApplicationContext@3c4297f: startup date [Sat Oct 19 18:22:05 CST 2019]; root of context hierarchy
2019-10-19 18:22:07.756 INFO 13820 --- [ main] o.s.b.w.embedded.tomcat.TomcatWebServer : Tomcat initialized with port(s): 8080 (http)
2019-10-19 18:22:07.870 INFO 13820 --- [ main] o.apache.catalina.core.StandardService : Starting service [Tomcat]
2019-10-19 18:22:07.870 INFO 13820 --- [ main] org.apache.catalina.core.StandardEngine : Starting Servlet Engine: Apache Tomcat/8.5.34
2019-10-19 18:22:07.896 INFO 13820 --- [ost-startStop-1] o.a.catalina.core.AprLifecycleListener : The APR based Apache Tomcat Native library which allows optimal performance in production environments was not found on the java.library.path: [D:\Program Files Java\Java\jdk1.8.0_162\bin;C:\Windows\Sun\Java\bin;C:\Windows\system32;C:\Windows;C:\ProgramData\Oracle\Java\javapath;C:\Windows\system32;C:\Windows;C:\Windows\System32\Wbem;C:\Windows\System32\WindowsPowerShell\v1.0\;D:\Program Files Java\SlikSvn\bin;D:\Program Files Java\MySQL Server 5.1\bin;D:\Program Files Java\TortoiseGit\bin;D:\Program Files\nodejs\;D:\Program Files Java\Java\jdk1.6.0_24\bin;D:\Program Files Java\apache-maven-3.2.3\bin;C:\Users\fuzhengwei\AppData\Roaming\npm;D:\Program Files Java\Git\cmd;;.]
2019-10-19 18:22:08.040 INFO 13820 --- [ost-startStop-1] o.a.c.c.C.[Tomcat].[localhost].[/] : Initializing Spring embedded WebApplicationContext
2019-10-19 18:22:08.040 INFO 13820 --- [ost-startStop-1] o.s.web.context.ContextLoader : Root WebApplicationContext: initialization completed in 2088 ms
2019-10-19 18:22:08.102 INFO 13820 --- [ost-startStop-1] o.s.b.w.servlet.ServletRegistrationBean : Servlet dispatcherServlet mapped to [/]
2019-10-19 18:22:08.126 INFO 13820 --- [ost-startStop-1] o.s.b.w.servlet.FilterRegistrationBean : Mapping filter: 'characterEncodingFilter' to: [/*]
2019-10-19 18:22:08.127 INFO 13820 --- [ost-startStop-1] o.s.b.w.servlet.FilterRegistrationBean : Mapping filter: 'hiddenHttpMethodFilter' to: [/*]
2019-10-19 18:22:08.127 INFO 13820 --- [ost-startStop-1] o.s.b.w.servlet.FilterRegistrationBean : Mapping filter: 'httpPutFormContentFilter' to: [/*]
2019-10-19 18:22:08.127 INFO 13820 --- [ost-startStop-1] o.s.b.w.servlet.FilterRegistrationBean : Mapping filter: 'requestContextFilter' to: [/*]
2019-10-19 18:22:09.118 INFO 13820 --- [ main] o.s.w.s.handler.SimpleUrlHandlerMapping : Mapped URL path [/**/favicon.ico] onto handler of type [class org.springframework.web.servlet.resource.ResourceHttpRequestHandler]
2019-10-19 18:22:09.383 INFO 13820 --- [ main] s.w.s.m.m.a.RequestMappingHandlerAdapter : Looking for @ControllerAdvice: org.springframework.boot.web.servlet.context.AnnotationConfigServletWebServerApplicationContext@3c4297f: startup date [Sat Oct 19 18:22:05 CST 2019]; root of context hierarchy
2019-10-19 18:22:10.261 INFO 13820 --- [ main] s.w.s.m.m.a.RequestMappingHandlerMapping : Mapped "{[/api/tree/decisionRuleTree],methods=[POST]}" onto public org.springframework.http.ResponseEntity org.itstack.demo.interfaces.DDDController.decisionRuleTree(org.itstack.demo.interfaces.dto.DecisionMatterDTO)
2019-10-19 18:22:10.263 INFO 13820 --- [ main] s.w.s.m.m.a.RequestMappingHandlerMapping : Mapped "{[/api/tree/queryTreeSummaryInfo],methods=[POST]}" onto public org.springframework.http.ResponseEntity org.itstack.demo.interfaces.DDDController.queryTreeSummaryInfo(org.itstack.demo.interfaces.dto.TreeDTO)
2019-10-19 18:22:10.272 INFO 13820 --- [ main] s.w.s.m.m.a.RequestMappingHandlerMapping : Mapped "{[/error]}" onto public org.springframework.http.ResponseEntity<java.util.Map<java.lang.String, java.lang.Object>> org.springframework.boot.autoconfigure.web.servlet.error.BasicErrorController.error(javax.servlet.http.HttpServletRequest)
2019-10-19 18:22:10.274 INFO 13820 --- [ main] s.w.s.m.m.a.RequestMappingHandlerMapping : Mapped "{[/error],produces=[text/html]}" onto public org.springframework.web.servlet.ModelAndView org.springframework.boot.autoconfigure.web.servlet.error.BasicErrorController.errorHtml(javax.servlet.http.HttpServletRequest,javax.servlet.http.HttpServletResponse)
2019-10-19 18:22:10.309 INFO 13820 --- [ main] o.s.w.s.handler.SimpleUrlHandlerMapping : Mapped URL path [/webjars/**] onto handler of type [class org.springframework.web.servlet.resource.ResourceHttpRequestHandler]
2019-10-19 18:22:10.309 INFO 13820 --- [ main] o.s.w.s.handler.SimpleUrlHandlerMapping : Mapped URL path [/**] onto handler of type [class org.springframework.web.servlet.resource.ResourceHttpRequestHandler]
2019-10-19 18:22:16.272 INFO 13820 --- [ main] o.s.j.e.a.AnnotationMBeanExporter : Registering beans for JMX exposure on startup
2019-10-19 18:22:16.273 INFO 13820 --- [ main] o.s.j.e.a.AnnotationMBeanExporter : Bean with name 'dataSource' has been autodetected for JMX exposure
2019-10-19 18:22:16.279 INFO 13820 --- [ main] o.s.j.e.a.AnnotationMBeanExporter : Located MBean 'dataSource': registering with JMX server as MBean [com.zaxxer.hikari:name=dataSource,type=HikariDataSource]
2019-10-19 18:22:16.375 INFO 13820 --- [ main] o.s.b.w.embedded.tomcat.TomcatWebServer : Tomcat started on port(s): 8080 (http) with context path ''
2019-10-19 18:22:16.381 INFO 13820 --- [ main] org.itstack.demo.DDDApplication : Started DDDApplication in 11.458 seconds (JVM running for 20.584)
2019-10-19 18:22:31.336 INFO 13820 --- [nio-8080-exec-1] o.a.c.c.C.[Tomcat].[localhost].[/] : Initializing Spring FrameworkServlet 'dispatcherServlet'
2019-10-19 18:22:31.336 INFO 13820 --- [nio-8080-exec-1] o.s.web.servlet.DispatcherServlet : FrameworkServlet 'dispatcherServlet': initialization started
2019-10-19 18:22:31.372 INFO 13820 --- [nio-8080-exec-1] o.s.web.servlet.DispatcherServlet : FrameworkServlet 'dispatcherServlet': initialization completed in 36 ms
2019-10-19 18:22:32.427 INFO 13820 --- [nio-8080-exec-1] o.itstack.demo.interfaces.DDDController : 規(guī)則樹行為信息決策10001Begin req:{"treeId":10001,"userId":"fuzhengwei","valMap":{"gender":"man","age":"25"}}
2019-10-19 18:22:32.508 INFO 13820 --- [nio-8080-exec-1] com.zaxxer.hikari.HikariDataSource : HikariPool-1 - Starting...
2019-10-19 18:22:32.956 INFO 13820 --- [nio-8080-exec-1] com.zaxxer.hikari.HikariDataSource : HikariPool-1 - Start completed.
2019-10-19 18:22:33.028 INFO 13820 --- [nio-8080-exec-1] o.i.d.d.rule.service.engine.EngineBase : 樹引擎=>Test分類規(guī)則樹 userId:fuzhengwei treeId:10001 treeNode:11 ruleKey:userGender matterValue:man
2019-10-19 18:22:33.028 INFO 13820 --- [nio-8080-exec-1] o.i.d.d.rule.service.engine.EngineBase : 樹引擎=>Test分類規(guī)則樹 userId:fuzhengwei treeId:10001 treeNode:112 ruleKey:userAge matterValue:25
2019-10-19 18:22:33.039 INFO 13820 --- [nio-8080-exec-1] o.itstack.demo.interfaces.DDDController : 規(guī)則樹行為信息決策10001End res:{"treeId":10001,"nodeValue":"果實B","success":true,"nodeId":112,"userId":"fuzhengwei"}
2019-10-19 18:23:36.989 INFO 13820 --- [nio-8080-exec-5] o.itstack.demo.interfaces.DDDController : 查詢規(guī)則樹信息10001Begin req:{"treeId":10001}
2019-10-19 18:23:37.006 INFO 13820 --- [nio-8080-exec-5] o.itstack.demo.interfaces.DDDController : 查詢規(guī)則樹信息10001End res:{"treeInfo":{"treeId":10001,"treeName":"購物分類規(guī)則樹","treeDesc":"用于分類不同類型用戶可購物范圍","nodeCount":7,"lineCount":6},"treeRulePointList":[{"ruleDesc":"用戶性別[男/女]","ruleKey":"userGender"},{"ruleDesc":"用戶年齡","ruleKey":"userAge"}]}

綜上總結(jié)

  • 以上模擬購物場景下的規(guī)則處理抽象為樹決策引擎,以達到獨立領(lǐng)域服務(wù)。另外決策服務(wù)可以使用drools,任何抽象并不一定永遠使用,不要拘泥于一種形式。
  • 一些大型架構(gòu)設(shè)計往往不是換一個設(shè)計模型就能徹底提升效率,還是需要人員整體素質(zhì),這是一個不斷培養(yǎng)的過程。
  • 領(lǐng)域驅(qū)動設(shè)計的思想并不只是教會程序猿寫代碼,也是非程序員以外的所有互聯(lián)網(wǎng)人員都適合學(xué)習(xí)的內(nèi)容。
  • 家里住的舒適不舒適,并不一定取決于三居或者四居,大部分還是依賴于怎么對格局的布置。事必躬親、親力親為的精益求精之路,終究會讓你設(shè)計出更加合理的代碼。
責(zé)任編輯:武曉燕 來源: 今日頭條
相關(guān)推薦

2021-09-08 09:22:23

領(lǐng)域驅(qū)動設(shè)計

2024-12-31 11:05:07

2024-11-27 15:33:17

軟件架構(gòu)DDD

2021-10-09 11:54:46

DDD微服務(wù)業(yè)務(wù)

2017-07-14 10:55:05

2020-09-02 08:12:05

CodeDDD代碼

2022-04-25 10:44:08

微服務(wù)架構(gòu)設(shè)計

2014-09-26 10:00:25

驅(qū)動設(shè)計DDD領(lǐng)域

2024-11-08 08:37:25

2010-07-14 09:01:07

架構(gòu)設(shè)計

2017-11-06 08:28:44

DDD架構(gòu)設(shè)計IT

2023-02-15 13:50:58

DDD戰(zhàn)略設(shè)計

2022-07-17 07:37:29

微服務(wù)DDD工程化落地

2023-04-13 08:23:28

軟件架構(gòu)設(shè)計

2024-04-17 08:06:41

六邊形洋蔥架構(gòu)領(lǐng)域

2021-06-07 09:33:21

企業(yè)架構(gòu)驅(qū)動

2024-07-17 08:12:06

2021-06-30 07:51:09

新項目領(lǐng)域建模

2022-04-11 09:15:00

前端開發(fā)技術(shù)

2024-09-25 08:00:00

領(lǐng)域驅(qū)動設(shè)計軟件開發(fā)
點贊
收藏

51CTO技術(shù)棧公眾號