【发布时间】:2015-06-19 14:44:24
【问题描述】:
我的问题很简单:如何把我班级的私有数据放到Repository中保存?
无论我们采用何种架构风格,每个人都同意业务对象不应该知道“如何”保存自己——也就是说,它们不应该实现数据库或其他持久性细节。但是,在我看来,业务对象是唯一知道他们需要保存“什么”的对象。存储库知道如何从数据库中获取数据,但如果它要知道如何将业务对象翻译成数据库术语,那么它必须知道要翻译的什么。
考虑到我可能使用数据库,但我没有用休眠注释标记我的类,因为我可能经常保存到纯文本文件。
假设我在这里的课程实际上是以特定的业务实体命名的,那么这样做会有什么问题
interface Exporter
{
public void Export(String id, String value1, String value2);
}
interface Repository
{
public Entity FindById(String id);
}
class Entity
{
private String id;
private String value1;
private String value2;
private String derivedvalue;
public Entity() {}
public Entity(String id, String value1, String value2)
{
this.id = id;
this.value1 = value1;
this.value2 = value2;
this.derivedvalue = /* derived from id, value1, and value2 */;
}
public void DoBusiness()
{
// ...
}
public void Export(Exporter exporter)
{
Exporter.Export(this.id, this.value1, this.value2);
}
}
并像使用它
FlatFileRepositoryAndExporter flatfile = new FlatFileRepositoryAndExporter(...);
Entity entity = flatfile.FindById(...);
// Do business logic
entity.DoBusiness();
entity.Export(flatfile);
我知道有一些框架可以帮助我,但归根结底,它们都依赖于某种反射。我想知道,在没有反射的情况下,我如何静态组合我的对象以在保持封装的同时公开它们的数据。我能想出的唯一答案就是这种访问者模式。
【问题讨论】:
-
这很好。我做过类似的事情,我将依赖项注入实体以执行各种操作(例如持久性),它对我来说效果很好。
-
访问者模式没问题,但它给域实体增加了一些噪音。您说得对,所有框架和持久性客户端都使用反射保存和检索对象。最简单的是 xml 或 json 序列化程序,无需任何映射即可工作。这同样适用于 NoSQL 数据库。将持久性逻辑添加到您的域将迫使您开始对其进行测试,这实际上并不是域测试的一部分。如果你想序列化到某个文件,你总是可以将你的存储库实现为一个通用列表,并使用一些序列化将它持久化到文件系统中。
标签: domain-driven-design encapsulation hexagonal-architecture