标题是 Repository,但是内容是我如何从错用的 Repository 变成了感觉还算对的 Repository 的过程。
DDD 里面的 Repository 是一个处理数据存储或者说是数据持久化的单元。通常一个 Aggregate 对应一个 Repository。对于通常的 web 服务,很多时候我们都是在与数据存储打交道,以至于很多时候存储就成为了真个应用最关键的逻辑了。那么刚刚接触 DDD 的时候,我就觉得 Repository 就是以前经常使用的类似 DAO 的东西。下面这样的代码经常出现在我们的应用里面。
final Application app = applicationRepository.create(name, user, stack);
return Response.created(routes.application(app)).build();其中 applicationRepository 管理了对于应用的创建。为了创建我们的对象 app 我们将一堆需要的参数扔进一个 applicationRepository 这样子的不知道背后是什么鬼实现的黑盒子,出来就是我们想要的东西了。再看另外一个例子。
public class ApplicationRecord implements Application, Record {
...
@Override
public void removeEnv(String envName) {
mapper.removeEnv(envName, this);
}
...
}在这里例子里面,application 可以有环境变量,在 application 中提供了一个 removeEnv 的方法,mapper 是一个具体的持久层工具 Mybatis 需要的东西,可以忽略。当我需要删除环境变量的时候,我只需要做如下的事情。
Application app = applicationRepository.ofId(appId);
app.removeEnv(envName);在这里事实上我根本没有显性的调用任何持久化方法,在 app 里面持久化就偷偷的帮我把事情做了。然后需要注意的是我的 Application 仅仅是一个接口,实现它的是一个 ApplicationRecord 它内部通过注入的方式塞进去了 Mybatis DataMapper 的东西从而实现了持久化的工作。然后在 Mybatis 可以放一个叫做 ObjectFactory 的东西使得 Mybatis 和 java injector 关联子一起,当从 Mybatis 获取对象时 Mybatis 会自动的讲所有的依赖注入到这个对象里。
说白了就是将数据层和模型绑定在一起,持久层做了业务层的事情。
然而我希望的是可以将业务层做成这个样子:
Repository 作为和存储打交道的组件应该仅仅是做持久化,它就是拿来一个对象,然后存到数据库里,没有任何业务逻辑,没有任何花哨的方法。interface ApplicationRepository {
void save(Application application);
Application ofId(String appId);
}没有什么 addEnv removeEnv 等等,这些都是 Application 自己要做的。Mybatis 版本的 Repository 具体的 mapper 仅仅出现在 MybatisApplicationRepository 里面,其他地方都不会出现。按照这个思路把上边的代码修改之后是下面这个样子。
final Application app = new Application(name, user, stack);
applicationRepository.save(app);
return Response.created(routes.application(app)).build();新创建的 app 本身就是一个 POJO 里面全部都是纯粹的业务代码。
public class Application {
private Envs envs;
...
public void removeEnv(String envName) {
envs.remove(envName);
}
...
}Application 有一个 envs 的属性,在调用 removeEnv 之后,application 的环境变量就更新了。如果需要持久化,就单独调用 applicationRepository。
Application app = applicationRepository.ofId(appId);
app.removeEnv(envName);
applicationRepository.save(app);这样的话持久化就和业务逻辑完全的分离开了,所有的 POJO 保证即使没有持久化也都可以正常的运转。领域对象是 class 而不是 interface 保证了内部的逻辑都是包含在业务层的。