浅谈Java开发架构之领域驱动设计DDD落地

网友投稿 242 2023-01-06

浅谈Java开发架构之领域驱动设计DDD落地

目录一、前言二、开发目标三、服务架构3.1、应用层{application}3.2、领域层{domain}3.3、基础层{infrastructrue}3.4、接口层{interfaces}四、开发环境五、代码示例六、综上总结

一、前言

整个过程大概是这样的,开发团队和领域专家一起通过 通用语言(Ubiquitous Language)去理解和消化领域知识,从领域知识中提取和划分为一个一个的子领域(核心子域,通用子域,支撑子域),并在子领域上建立模型,再重复以上步骤,这样周而复始,构建出一套符合当前领域的模型。

二、开发目标

依靠领域驱动设计的设计思想,通过事件风暴建立领域模型,合理划分领域逻辑和物理边界,建立领域对象及服务矩阵和服务架构图,定义符合DDD分层架构思想的代码结构模型,保证业务模型与代码模型的一致性。通过上述设计思想、方法和过程,指导团队按照DDD设计思想完成微服务设计和开发。

1、拒绝泥球小单体、拒绝污染功能与服务、拒绝一加功能排期一个月

2、架构出高可用极易符合互联网高速迭代的应用服务

3、物料化、组装化、可编排的服务,提高人效

三、服务架构

3.1、应用层{application}

应用服务位于应用层。用来表述应用和用户行为,负责服务的组合、编排和转发,负责处理业务用例的执行顺序以及结果的拼装。

应用层的服务包括应用服务和领域事件相关服务。

应用服务可对微服务内的领域服务以及微服务外的应用服务进行组合和编排,或者对基础层如文件、缓存等数据直接操作形成应用服务,对外提供粗粒度的服务。

领域事件服务包括两类:领域事件的发布和订阅。通过事件总线和消息队列实现异步数据传输,实现微服务之间的解耦。

3.2、领域层{domain}

领域服务位于领域层,为完成领域中跨实体或值对象的操作转换而封装的服务,领域服务以与实体和值对象相同的方式参与实施过程。

领域服务对同一个实体的一个或多个方法进行组合和封装,或对多个不同实体的操作进行组合或编排,对外暴露成领域服务。领域服务封装了核心的业务逻辑。实体自身的行为在实体类内部实现,向上封装成领域服务暴露。

为隐藏领域层的业务逻辑实现,所有领域方法和服务等均须通过领域服务对外暴露。

为实现微服务内聚合之间的解耦,原则上禁止跨聚合的领域服务调用和跨聚合的数据相互关联。

3.3、基础层{infrastructrue}

基础服务位于基础层。为各层提供资源服务(如数据库、缓存等),实现各层的解耦,降低外部资源变化对业务逻辑的影响。

基础服务主要为仓储服务,通过依赖反转的方式为各层提供基础资源服务,领域服务和应用服务调用仓储服务接口,利用仓储实现持久化数据对象或直接访问基础资源。

3.4、接口层{interfaces}

接口服务位于用户接口层,用于处理用户发送的Restful请求和解析用户输入的配置文件等,并将信息传递给应用层。

四、开发环境

jdk1.8【jdk1.7以下只能部分支持netty】

springboot 2.0.6.RELEASE

idea + maven

五、代码示例

itstack-demo-ddd-01

└── src

    ├── main

    │   ├── java

    │PMiSubcVB160;  │   └── org.itstack.demo

    │   │       ├── application

    │   │       │ ├── event

    │   │       │ │   └── ApplicationRunner.java

    │   │       │ └── service

    │   │       │     └── UserService.java

    │   │       ├── domain

http://    │   │       │ ├── model

    │   │       │ │   ├── aggregates

    │   │       │ │http://60;  │   └── UserRichInfo.java

    │   │       │ │   └── vo

    │   │       │ │       ├── UserInfo.java

    │   │       │ │       └── UserSchool.java

    │   │       │ ├── repository

    │   │       │ │   └── IuserRepository.java

    │   │       │ └── service

    │   │       │     └── UserServiceImpl.java

    │   │       ├── infrastructure

    │   │       │ ├── dao

    │   │       │ │   ├── impl

    │   │       │ │   │   └── UserDaoImpl.java

    │   │       │ │   └── UserDao.java

    │   │       │ ├── po

    │   │       │ │   └── UserEntity.java

    │   │       │ ├── repository

    │   │       │ │   ├── mysql

    │   │       │ │   │   └── UserMysqlRepository.java

    │   │       │ │   ├── redis

    │   │       │ │   │   └── UserRedisRepository.java

    │   │       │ │   └── UserRepository.java

    │   │       │ └── util

    │   │       │     └── RdisUtil.java

    │   │       ├── interfaces

    │   │       │ ├── dto

    │   │       │ │ └── UserInfoDto.java

    │   │       │ └── facade

    │   │       │ └── DDDController.java

    │   │       └── DDDApplication.java

    │   ├── resources

    │   │   └── application.yml

    │   └── webapp

    │       └── WEB-INF

    │        └── index.jsp

    └── test

         └── java

             └── org.itstack.demo.test

                 └── ApiTest.java

application/UserService.java | 应用层用户服务,领域层服务做具体实现

public interface UserService {

UserRichInfo queryUserInfoById(Long id);

}

domain/repository/IuserRepository.java | 领域层资源库,由基础层实现

public interface IUserRepository {

void save(UserEntity userEntity);

UserEntity query(Long id);

}

domain/service/UserServiceImpl.java | 应用层实现类,应用层是很薄的一层可以只做服务编排

@Service("userService")

public class UserServiceImpl implements UserService {

@Resource(name = "userRepository")

private IUserRepository userRepository;

@Override

public UserRichInfo queryUserInfoById(Long id) {

// 查询资源库

UserEntity userEntity = userRepository.query(id);

UserInfo userInfo = new UserInfo();

http:// userInfo.setName(userEntity.getName());

// TODO 查询学校信息,外部接口

UserSchool userSchool_01 = new UserSchool();

userSchool_01.setSchoolName("振华高级实验中学");

UserSchool userSchool_02 = new UserSchool();

userSchool_02.setSchoolName("东北电力大学");

List userSchoolList = new ArrayList<>();

userSchoolList.add(userSchool_01);

userSchoolList.add(userSchool_02);

UserRichInfo userRichInfo = new UserRichInfo();

userRichInfo.setUserInfo(userInfo);

userRichInfo.setUserSchoolList(userSchoolList);

return userRichInfo;

}

}

infrastructure/po/UserEntity.java | 数据库对象类

public class UserEntity {

private Long id;

private String name;

get/set ...

}

infrastructrue/repository/UserRepository.java | 领域层定义接口,基础层资源库实现

@Repository("userRepository")

public class UserRepository implements IUserRepository {

@Resource(name = "userMysqlRepository")

private IUserRepository userMysqlRepository;

@Resource(name = "userRedisRepository")

private IUserRepository userRedisRepository;

@Override

public void save(UserEntity userEntity) {

//保存到DB

userMysqlRepository.save(userEntity);

//保存到Redis

userRedisRepository.save(userEntity);

}

@Override

public UserEntity query(Long id) {

UserEntity userEntityRedis = userRedisRepository.query(id);

if (null != userEntityRedis) return userEntityRedis;

UserEntity userEntityMysql = userMysqlRepository.query(id);

if (null != userEntityMysql){

//保存到Redis

userRedisRepository.save(userEntityMysql);

return userEntityMysql;

}

// 查询为NULL

return null;

}

}

interfaces/dto/UserInfoDto.java | DTO对象类,隔离数据库类

public class UserInfoDto {

private Long id; // ID

public Long getId() {

return id;

}

public void setId(Long id) {

this.id = id;

}

}

interfaces/facade/DDDController.java | 门面接口

@Controller

public class DDDController {

@Resource(name = "userService")

private UserService userService;

@RequestMapping("/index")

public String index(Model model) {

return "index";

}

@RequestMapping("/api/user/queryUserInfo")

@ResponseBody

public ResponseEntity queryUserInfo(@RequestBody UserInfoDto request) {

return new ResponseEntity<>(userService.queryUserInfoById(request.getId()), HttpStatus.OK);

}

}

六、综上总结

以上基于DDD一个基本入门的结构演示完成,实际开发可以按照此模式进行调整。目前这个架构分层还不能很好的进行分离,以及层级关系的引用还不利于扩展。

以上就是浅谈Java开发架构之领域驱动设计DDD落地的详细内容,更多关于Java开发架构 领域驱动设计DDD落地的资料请关注我们其它相关文章!

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:在SpringBoot中,如何使用Netty实现远程调用方法总结
下一篇:青海快递物流查询单号电话(西海物流快递单号查询)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~