user-interface – 用户界面如何知道允许哪些命令对聚合根执行?
UI与域分离,但UI应尽力永远不允许用户发出肯定会失败的命令.
考虑以下示例(伪代码): DiscussionController @Security(is_logged) @Method('POST') @Route('addPost') addPostToDiscussionAction(request) discussionService.postToDiscussion( new PostToDiscussionCommand(request.discussionId,session.myUserId,request.bodyText) ) @Method('GET') @Route('showDiscussion/{discussionId}') showDiscussionAction(request) discussionWithAllThePosts = discussionFinder.findById(request.discussionId) canAddPostToThisDiscussion = ??? // render the discussion to the user,and use `canAddPostToThisDiscussion` to show/hide the form // from which the user can send a request to `addPostToDiscussionAction`. renderDiscussion(discussionWithAllThePosts,canAddPostToThisDiscussion) PostToDiscussionCommand constructor(discussionId,authorId,bodyText) DiscussionApplicationService postToDiscussion(command) discussion = discussionRepository.get(command.discussionId) author = collaboratorService.authorFrom(discussion.Id,command.authorId) post = discussion.createPost(postRepository.nextIdentity(),author,command.bodyText) postRepository.add(post) DiscussionAggregate // originalPoster is the Author that started the discussion constructor(discussionId,originalPoster) // if the discussion is closed,you can't create a post. // *unless* if you're the author (OP) that started the discussion createPost(postId,bodyText) if (this.close && !this.originalPoster.equals(author)) throw "Discussion is closed." return new Post(this.discussionId,postId,bodyText) close() if (this.close) throw "Discussion already closed." this.close = true isClosed() return this.close >用户转到/ showDiscussion / 123,他看到与< form>的讨论.他可以从中提交新帖子,但仅在讨论未结束或当前用户是谁开始讨论时才提交. 如何将这些知识提供给UI? 我可以将其编码到读取模型中, canAddPostToThisDiscussion = !discussionWithAllThePosts.discussion.isClosed && discussionWithAllThePosts.discussion.originalPoster.id == session.currentUserId 但后来我需要维护这个逻辑并使其与写模型保持同步.这是一个相当简单的例子,但随着聚合的状态转换变得更加复杂,它可能变得非常困难.我想将我的聚合图像作为状态机,以及它们的工作流程(如RESTBucks示例).但我不喜欢将业务逻辑移到我的域模型之外的想法,并把它放在读取端和写入端都可以使用的服务中. 也许这不是最好的例子,但由于聚合根基本上是一致性边界,我们知道我们需要在其生命周期中防止无效状态转换,并且在每次转换到新状态时,某些操作可能变为非法和副反之亦然.那么,用户界面如何知道允许与否?我有什么选择?我该如何处理这个问题?你有什么例子可以提供吗? 解决方法
最简单的方法可能是分享域模型对UI可能性的理解.塔达 这是一种思考它的方法 – 在抽象中,所有写模型逻辑都具有相当简单的外观形状. { // Notice that these statements are queries State currentState = bookOfRecord.getState() State nextState = model.computeNextState(currentState,command) // This statement is a command bookOfRecord.replace(currentState,nextState) } 这里的关键思想:记录册是国家权威;其他人(包括“写模型”)正在使用陈旧的副本. 模型表示的是一组约束,可确保满足业务不变性.在系统的生命周期中,可能存在许多不同的约束集,因为对业务的理解会发生变化. 写入模型是在替换记录簿中的状态时当前强制执行约束的权限.其他人都在使用陈旧的副本. 陈旧是要牢记的;在分布式系统中,您执行的任何验证都是临时的 – 除非您锁定状态并锁定模型,否则可以在邮件处于运行状态时更改. 这意味着您的验证无论如何都是近似的,因此您不必过于担心您的所有细节都是正确的.您假设状态的陈旧副本大致正确,并且您当前对模型的理解大致正确,并且如果命令在给定这些前置条件的情况下有效,则检查它是否足以发送.
我认为这里最好的答案是“克服它”.我知道了;因为在聚合根中包含业务逻辑是文献告诉我们要做的事情.但是如果你继续重构,识别常见模式并分离问题,你会发现实体实际上只是围绕对状态和functional core的引用进行管道调整. AggregateRoot { final Reference<State> bookOfRecord; final Model<State,Command> theModel; onCommand(Command command) { State currentState = bookOfRecord.getState() State nextState = model.computeNextState(currentState,command) bookOfRecord.replace(currentState,nextState) } } 我们在这里所做的就是采用“构造下一个状态”逻辑,我们过去常常将其分散到AggregateRoot中,并将其封装到一个单独的责任边界中.这里,它特定于根本身,但是等效的重构它将它作为参数传递. AggregateRoot { final Reference<State> bookOfRecord; onCommand(Model<State,Command> theModel,Command command) { State currentState = bookOfRecord.getState() State nextState = model.computeNextState(currentState,nextState) } } 换句话说,从跟踪状态的管道中取出的模型是域服务.域服务中的域逻辑与聚合中的域逻辑一样,也是域模型的一部分 – 两个实现是彼此双重的. 并且您的域的读取模型没有理由不能访问域服务. (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |