- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
场景如下:
我加入了一个团队,该团队正开始开发一个相当大的金融系统,并且据说以 DDD 为目标,具有丰富的行为领域模型。但是,我可以在项目的当前区域看到许多指向 Active Record 的气味:应用程序服务中的单个“Save”方法;一组相当贫乏的领域对象,它们有一个大的“验证”方法等。所有这些都非常像 CRUD,并且变得越来越难以管理。
我正在开发的用户故事大致翻译为“当直接借记从特定状态转变为另一种状态(未预算到预算)时,创建一个任务项目供顾问处理”。有多种规则可以确定“未列入预算”和“已列入预算”的含义。
目前 DirectDebit 类根本没有“状态”的概念,它是一个相当大的 DTO(有几个子对象——同样,所有 DTO),有大约 1000 行验证逻辑来尝试确定什么它应该处于的形状。企业对直接借记可以处于的不同状态有“清晰”的概念,我相信该领域足够丰富,可以保证领域模型。
DirectDebit 应用程序服务有一个“保存”方法,它传递更新后的直接借记(从 UI)。该方法从存储库中获取现有的 DirectDebit,完全忽略它,对更新的直接借记对象运行“验证”,并将其保存到存储库中。然后该方法审核新旧直接借记对象之间的变化。
我正试图摆脱这里开始发展的类似 CRUD 的怪物,并向团队展示他们如何开始重构它。
理想情况下,我想开始重构领域对象以使用状态模式,使用良好的封装,也许使用领域事件来冒泡状态已经改变的事实。应用层将有一个名为“CreateTaskWhenBudgeted”或类似名称的领域事件处理程序来解耦逻辑。
我的问题是来自 UI 的大型“保存”直接借记方法。
是否有分解/重构这种方法的策略?没有简单的方法来判断什么是“改变的”,因为没有明确的命令正在发生。或者,我是否应该继续增加困惑而忽略我认为是问题的东西......?
这是应用程序服务的简化版本:
public void SaveDirectDebit(DirectDebit directDebit)
{
directDebit.Validate(false);
var beforeDirectDebit = _directDebitRepository.FetchDirectDebit(directDebit.ExpenditureId.Id);
_directDebitRepository.SaveDirectDebit(directDebit);
var afterDirectDebit = _directDebitRepository.FetchDirectDebit(directDebit.ExpenditureId.Id);
var accountAuditLog = CreateDirectDebitAudit(beforeDirectDebit, afterDirectDebit);
_auditLogRepository.CreateAudit(accountAuditLog);
}
最佳答案
当然没有一种方法可以做到(好方法与其他方法),您有多个参数(时间、复杂性...)。
由内而外的方式
免责声明:遗留代码越庞大,这种方法似乎就越难以实现(对我来说成功的条件:你不应该是唯一关心这种重构的人)。
-> 保存到数据库是个技术活儿。你必须首先引入一些通用语言来使隐含显式,并且能够引入概念和进行设计(创建边界和分离上下文)。
另一种方式
这是我在评论中开始描述的另一种方式,对我来说,当你被遗留系统包围时,它与 Eric Evans 所说的“气泡上下文”有关,从一个干净的小有界上下文开始会更容易。这种隔离将帮助您应用比现有系统更好(不同)的设计。
关于c# - 从 Active Record 转移到领域对象模式方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34856669/
如果我将我的个人 repo 转移到一个组织(由我创建),我将失去所有 见解 例如来自原始 Repo 的流量历史记录、拉取请求、贡献者、 fork 等? 最佳答案 拉取请求被保留:参见“About re
如何为解析 if-then[-else] 案例制定正确的规则?这是一些语法: { module TestGram (tparse) where } %tokentype { String }
如何为解析 if-then[-else] 案例制定正确的规则?这是一些语法: { module TestGram (tparse) where } %tokentype { String }
我读过有关mutex的信息,这些信息由线程拥有,并且只能由拥有的线程使用。在this answer中,该解决方案建议每个进程在发出互斥信号之前,必须拥有互斥锁的所有权。我必须在这里承认自己的愚蠢,不知
我只能从回调函数之一中想到 curl_close() 。 但是 php 抛出了一个警告: PHP 警告:curl_close():尝试从回调中关闭 cURL 句柄。 任何想法如何做到这一点? 最佳答案
带有冲突的语法的精简版本: body: variable_list function_list; variable_list: variable_list variable | /* empty
我创建了新的开发者帐户,然后将应用程序转移到新帐户。然后我在新帐户下创建了相同的标识符。并构建App并上传到AppStore。 I have got the warning with WARNING
我想像这样管理类主任的所有 Activity : 此外所有 Activity 都扩展基本 Activity 以使用公共(public) View 。 在这种情况下,我想处理传输 Activity ,例
使用 C 中的简单链表实现,我如何告诉 Splint 我正在转让 data 的所有权? typedef struct { void* data; /*@null@*/ void* ne
请参阅以下 yacc 代码。如果我删除生产因素:'!' expr,解析冲突消失。这里发生了什么? %{ #include #include %} %token TRUE %token FALSE
是否可以将 props 向下传输到子组件,其中 { ..this.props } 用于更简洁的语法,但是排除某些 props,如 className 或 id? 最佳答案 您可以使用解构来完成这项工作
如果我有以下数据框: date A B M S 20150101 8 7 7.5 0 20150101 10 9 9
我需要将一个 __m128i 变量(比如 v)移动 m 位,以便位移动所有变量(因此,结果变量表示 v*2^m)。执行此操作的最佳方法是什么?! 请注意 _mm_slli_epi64 分别移动 v0
我需要这样调用我的程序: ./program hello -r foo bar 我从 argv[1] 中打招呼,但我在使用值 bar 时遇到问题,我是否也应该将“r:”更改为其他内容? while((
我是新来的 Bison我在转换/减少冲突方面遇到了麻烦...我正在尝试从文件加载到 array data[] : struct _data { char name[50]; char sur
当然有很多关于解决移位/归约错误的文档和方法。 Bison 文档建议正确的解决方案通常是%期待它们并处理它。 当你遇到这样的事情时: S: S 'b' S | 't' 您可以像这样轻松解决它们: S:
我有以下(大量精简的)快乐语法 %token '{' { Langle } '}' { Rangle } '..' { DotDot } '::' { ColonC
我的 Bison 解析器中有很多错误,即使它运行良好,我也想了解这些冲突。代码如下: 词法分析器: id ([[:alpha:]]|_)([[:alnum:]]|_)* %% {id
在我的项目中,我有这样的情况,一个 Activity 应该将值(value)转移到另一个 Activity 。并且根据这个值应该选择需要的菜单元素。我试图在 bundle 的帮助下做到这一点,但我不知
我一直在阅读 NSIndexPaths 以获得 uitableviews 等。但是我很难操纵现有的索引路径。 我想在保留行的同时采用现有的索引路径递增/移动每个部分。因此 indexPath.sect
我是一名优秀的程序员,十分优秀!