分布领域驱动设计(DDD):领域接口化设计式缓存的选择
架构之美
67篇原创内容
公众号
- 前言 -
把服务对象(service)和资源库对象(repository)设计成接口是最常见的。但是这对接口化的认识还远远不够,我们需要更深入地去分析接口化设计和更全面地应用接口化编程。所以我们要讨论的是全面接口化,尤其是对领域模型接口化的认识。
- 领域接口化 -
通常的情况下我们会把领域模型设计成类(class),但是你有没有想过把领域模型设计成接口(interface)?比如:
public interface User {
// ...
}
public class UserImpl implements User {
// ...
}
这样的设计似乎没有任何价值,那么继续深入地看看。比如:
public class JpaUserRepository implements UserRepository {
// ...
@Override
public Optional<User> findById(String id) {
UserPO userPO = this.entityManager.find(UserPO.class, id);
return Optional.ofNullable(userPO).map(UserPO::toUser);
}
@Override
public User save(User user) {
UserPO userPO = this.entityManager.find(UserPO.class, user.getId());
userPO.setNickname(user.getNickname());
// ...
return this.entityManager.merge(userPO).toUser();
}
}
public class JpaUserRepository implements UserRepository {
补充 JpaUser.of() 方法的实现:
public class JpaUser extends UserSupport { // ... public static JpaUser of(User user) { if (user instanceof JpaUser) { return (JpaUser) user; } var target = new JpaUser(); BeanUtils.copyProperties(user, target); // ... return target; }}
public interface ElasticsearchUserRepository
extends ElasticsearchRepository<ElasticsearchUser, String> {
// extends ElasticsearchRepository<User, String> // Not supported
}
public class DelegatingElasticsearchUserRepository implements UserRepository {
private final ElasticsearchUserRepository elasticsearchUserRepository;
public DelegatingElasticsearchUserRepository(ElasticsearchUserRepository elasticsearchUserRepository) {
this.elasticsearchUserRepository = elasticsearchUserRepository;
}
@Override
public User create(String id) {
return new ElasticsearchUser(id);
}
@Override
public Optional<User> findById(String id) {
return CastUtils.cast(this.elasticsearchUserRepository.findById(id));
}
@Override
public User save(User user) {
return this.elasticsearchUserRepository.save(ElasticsearchUser.of(user));
}
// ...
}
- 关联接口化 -
@Getter@Setter@NoArgsConstructor@Entity@Table(name = 'mf_order')public class JpaOrder implements Order { // ... // OrderItem 是一个接口类型,不能持久化。 private List<OrderItem> items = new ArrayList<>(); // ...}
// ...
public class JpaOrder implements Order {
// ...
// 通过指定具体的 targetEntity 类型,来解决泛化与特化的问题。
@OneToMany(targetEntity = JpaOrderItem.class)
private List<OrderItem> items = new ArrayList<>();
// ...
}
支持 targetEntity 属性的注解包括:@OneToMany、@OneToOne、@ManyToOne、@ManyToMany。
@Getter
@Setter
@NoArgsConstructor
@Document(indexName = 'user')
public class ElasticsearchOrder implements Order {
// ...
// 使用具体特化类型进行解决。
private List<ElasticsearchOrderItem> items = new ArrayList<>();
@Override
public void setItems(List<OrderItem> items) {
this.items = Objects.requireNonNullElseGet(items, (Supplier<List<OrderItem>>) ArrayList::new)
.stream().map(ElasticsearchOrderItem::of).collect(Collectors.toList());
}
// ...
}
<resultMap id='Order' type='org.mallfoundry.order.repository.mybatis.MybatisOrder'> <!-- ... --> <collection property='items' ofType='org.mallfoundry.order.repository.mybatis.MybatisOrderItem'> <!-- ... --> </collection> <!-- ... --></resultMap>
@Test
public void testCreateUser() {
User user = this.userService.createUser(null); // new User()
user.setNickname('Nickname');
user.setGender(Gender.MALE);
this.userService.addUser(user);
}
- 系统接口化 -
对于一个产品我们要考虑的不只是产品本身能解决的业务需求,还需要在部署上有所追求。如果项目初期的并发量很小,客户可能采用单进程的方式部署,慢慢地单进程扛不住了会升级到集群的方式,最终还要升级到微服务的方式。如何在单进程、集群和微服务之间进行无缝切换呢?
- 开源电商 -
Mallfoundry 是一个完全开源的使用 Spring Boot 开发的多商户电商平台。它可以嵌入到已有的 Java 程序中,或者作为服务器、集群、云中的服务运行。
领域模型采用领域驱动设计(DDD)、接口化以及面向对象设计。
- 总结 -
作者:不够具体
来源:https://juejin.cn/post/6894109393173315597
赞 (0)