分层的本质是关注点分离,隔离对下层的变化,可以简化复杂性,使得层次结构更加清晰。

1. 主流分层结构介绍

目前业界存在两种主流的应用工程结构:一种是阿里推出的《Java开发手册》中推荐的,另外一种是基于DDD(领域驱动设计)推荐的。

1.1 基于阿里《Java开发手册》的分层结构

• 开放 API 层:可直接封装 Service 接口暴露成 RPC 接口;通过 Web 封装成 http 接口;网关控制层等。
• 终端显示层:各个端的模板渲染并执行显示的层。当前主要是 velocity 渲染,JS 渲染,JSP 渲染,移
动端展示等。
• Web 层:主要是对访问控制进行转发,各类基本参数校验,或者不复用的业务简单处理等。
• Service 层:相对具体的业务逻辑服务层。
• Manager 层:通用业务处理层,它有如下特征:
1) 对第三方平台封装的层,预处理返回结果及转化异常信息,适配上层接口。
2) 对 Service 层通用能力的下沉,如缓存方案、中间件通用处理。 3) 与 DAO 层交互,对多个 DAO 的组合复用。
• DAO 层:数据访问层,与底层 MySQL、Oracle、Hbase、OB 等进行数据交互。
• 第三方服务:包括其它部门 RPC 服务接口,基础平台,其它公司的 HTTP 接口,如淘宝开放平台、支
付宝付款服务、高德地图服务等。
• 外部数据接口:外部(应用)数据存储服务提供的接口,多见于数据迁移场景中。

1.2 基于DDD(领域驱动设计)的分层结构

• 领域层:体现业务逻辑。
• 应用层:依赖领域层,根据业务对下层领域进行聚合和编排。
• 基础设施层:为其他提供技术支持。
• 用户接口层:为外部用户访问底层系统提供交互界面和数据表示。

2. 自己的工程结构

基于上述两种工程结构,设计一个适合自己的Java项目分层结构。

example
└─src
├─main
│ ├─java
│ │ └─com
│ │ └─example
│ │ ├─application --应用层(聚合多个领域)
│ │ ├─domain --领域层
│ │ │ ├─order --订单域
│ │ │ │ ├─bo --业务对象
│ │ │ │ ├─constant --领域内局部常量
│ │ │ │ ├─controller --控制器
│ │ │ │ ├─dto --数据传输对象
│ │ │ │ ├─event --事件
│ │ │ │ │ ├─publish --发布
│ │ │ │ │ └─subscribe --订阅
│ │ │ │ ├─manager --通用逻辑处理
│ │ │ │ ├─repository --存储
│ │ │ │ │ ├─entity --实体,对应数据库中的字段
│ │ │ │ │ └─mapper --mybatis mapper
│ │ │ │ └─service --业务层处理
│ │ │ │ └─impl --业务接口实现
│ │ │ └─user --用户域
│ │ │ ├─bo
│ │ │ ├─constant
│ │ │ ├─controller
│ │ │ ├─dto
│ │ │ ├─event
│ │ │ │ ├─publish
│ │ │ │ └─subscribe
│ │ │ ├─manager
│ │ │ ├─repository
│ │ │ │ ├─entity
│ │ │ │ └─mapper
│ │ │ └─service
│ │ │ └─impl
│ │ └─infrastructure --基础设施层
│ │ ├─config --配置
│ │ ├─constant --全局常量
│ │ ├─handler --处理器
│ │ ├─interceptor --拦截器
│ │ ├─thirdparty --第三方
│ │ └─utils --工具类
│ └─resources
│ ├─mapper
│ │ ├─order
│ │ └─user



└─test
└─java
└─com
└─example
  • 接收参数和响应报文,请求以Req为后缀,响应以Resp为后缀,代码写在dto包中,比如创建订单请求和响应
/**
* 创建订单请求
*/
@Data
public class OrderCreateReq { /**
* 用户id
*/
private String userId; /**
* 订单金额
*/
private BigDecimal amount; /**
* 下单的商品集合
*/
private List<OrderDetailReq> orderDetailReqList; @Data
public static class OrderDetailReq { /**
* 商品id
*/
private Long goodsId;
/**
* 商品数量
*/
private Integer goodsNum; }
} /**
* 创建订单响应
*/
@Data
public class OrderCreateResp { /**
* 订单id
*/
private String orderId;
}
  • DAO层代码放在repository中

  • 业务层代码放在service和manager中,比如创建订单因为涉及到订单表和订单明细表,需要在一个事务中,所以将事务代码下沉到manager。

@Service
public class OrderServiceImpl implements OrderService { @Resource
private OrderManager orderManager; @Override
public OrderCreateResp create(OrderCreateReq req) { Order order = buildOrder(req);
List<OrderDetail> orderDetailList = buildOrderDetailList(order.getOrderId(), req); orderManager.createOrder(order, orderDetailList); OrderCreateResp resp = new OrderCreateResp();
resp.setOrderId(order.getOrderId());
return resp;
} private Order buildOrder(OrderCreateReq req) {
Order order = new Order();
order.setOrderId(UUID.randomUUID().toString());
order.setUserId(req.getUserId());
order.setAmount(req.getAmount());
return order;
} private List<OrderDetail> buildOrderDetailList(String orderId, OrderCreateReq req) {
List<OrderDetail> orderDetailList = new ArrayList<>();
for (OrderCreateReq.OrderDetailReq orderDetailReq : req.getOrderDetailReqList()) {
OrderDetail orderDetail = new OrderDetail();
orderDetail.setOrderId(orderId);
orderDetail.setGoodsId(orderDetailReq.getGoodsId());
orderDetail.setGoodsNum(orderDetailReq.getGoodsNum());
orderDetailList.add(orderDetail);
}
return orderDetailList;
}
}
@Component
public class OrderManager { @Resource
private OrderMapper orderMapper; @Resource
private OrderDetailMapper orderDetailMapper; @Transactional(propagation = Propagation.REQUIRED, rollbackFor = Exception.class)
public void createOrder(Order order, List<OrderDetail> orderDetailList) {
orderMapper.insert(order);
for (OrderDetail orderDetail : orderDetailList) {
orderDetailMapper.insert(orderDetail);
}
} }
  • 业务对象存放在bo包中,比如查询用户信息,不需要返回密码字段,则可以定义一个UserBO。

@Data
public class UserBO { private String userId; private String username; private String nickname;
}
  • application层做聚合编排,比如下单,既要保存订单信息,又要扣减库存,就需要对订单域和库存域进行聚合编排。

Java应用工程结构的更多相关文章

  1. Java Web工程目录结构

    说明 介绍java web 工程的基本结构目录.记录java web 工程结构的学习. Java web 结构 java web严格来说分为两类工程结构:一个是工程编译目录结构,一个是工程发布目录结构 ...

  2. [转]Java Web工程目录结构

    说明 介绍java web 工程的基本结构目录.记录java web 工程结构的学习. Java web 结构 java web严格来说分为两类工程结构:一个是工程编译目录结构,一个是工程发布目录结构 ...

  3. 重学 Java 设计模式:实战装饰器模式(SSO单点登录功能扩展,增加拦截用户访问方法范围场景)

    作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言 对于代码你有编程感觉吗 很多人写代码往往是没有编程感觉的,也就是除了可以把功能按照固 ...

  4. 重学 Java 设计模式:实战享元模式「基于Redis秒杀,提供活动与库存信息查询场景」

    作者:小傅哥 博客:https://bugstack.cn 沉淀.分享.成长,让自己和他人都能有所收获! 一.前言 程序员‍‍的上下文是什么? 很多时候一大部分编程开发的人员都只是关注于功能的实现,只 ...

  5. 基于IDEA 最新Spirng3.2+hibernate4+struts2.3 全注解配置 登录

    原文 基于IDEA 最新Spirng3.2+hibernate4+struts2.3 全注解配置 登录 首先说说 IDEA 12,由于myeclipse越来越卡,我改用idea12 了,发现其功能强悍 ...

  6. Hadoop2源码分析-准备篇

    1.概述 我们已经能够搭建一个高可用的Hadoop平台了,也熟悉并掌握了一个项目在Hadoop平台下的开发流程,基于Hadoop的一些套件我们也能够使用,并且能利用这些套件进行一些任务的开发.在Had ...

  7. org.thymeleaf.exceptions.TemplateInputException: Error resolving template "/ template might not exist or might not be accessible by any of the configured

    异常现象:在本地打包部署完全没有问题,资源文件也都可以映射上,但是打包成jar包部署到服务器上时,就一直报异常,异常信息如下: 严重: Servlet.service() for servlet [d ...

  8. Spring Cloud feign

    Spring Cloud feign使用 前言 环境准备 应用模块 应用程序 应用启动 feign特性 综上 1. 前言 我们在前一篇文章中讲了一些我使用过的一些http的框架 服务间通信之Http框 ...

  9. flex与j2ee的结合(flex+Spring)

    分类: flex spring2012-04-25 02:11 1262人阅读 评论(1) 收藏 举报 flexspringactionscriptjavapropertiesservlet   目录 ...

随机推荐

  1. Linux-centos7安装mysql步骤

    原文地址:https://blog.csdn.net/qq_42467339/article/details/89471395 Centos7.3 yum安装MySQL5.7.25 扩展:在CentO ...

  2. Mybatis的ResultMap对column和property

    首先,先看看这张图,看能不能一下看明白:   select元素有很多属性(这里说用的比较多的): id:命名空间唯一标识,可以被用来引用这条语句 parameterType:将会传入这条语句的参数类的 ...

  3. 恢复mysql管理员密码

    1.向mysqld server 发送kill命令关掉mysqld server(不是 kill -9),存放进程ID的文件通常在MYSQL的数据库所在的目录/var/lib/mysql中.# kil ...

  4. dedecms 5.7 任意前台用户修改漏洞

    一. 启动环境 1.双击运行桌面phpstudy.exe软件 2.点击启动按钮,启动服务器环境 二.代码审计 1.双击启动桌面Seay源代码审计系统软件 2.点击新建项目按钮,弹出对画框中选择(C:\ ...

  5. position 的值absolute、fixed、relative和static的定位原点是什么

    position 的值absolute.fixed.relative和static的定位原点是什么 absolute 成绝对定位的元素,相对于值不为static的第一个父元素进行定位,也可以理解为离自 ...

  6. SpringBoot bean映射yml中的属性举例

    pom:导入配置文件处理器,配置文件进行绑定就会有提示 <dependency> <groupId>org.springframework.boot</groupId&g ...

  7. java-规约-日期和时间

    public class DateTime { public static void main(String[] args) { /**1 * 日期格式化时,传入的pattern表示年份统一用小写的y ...

  8. Redis 集群方案什么情况下会导致整个集群不可用?

    有 A,B,C 三个节点的集群,在没有复制模型的情况下,如果节点 B 失败了, 那么整个集群就会以为缺少 5501-11000 这个范围的槽而不可用.

  9. 如何通过HibernateDaoSupport将Spring和Hibernate 结合起来?

    用 Spring 的 SessionFactory 调用 LocalSessionFactory.集成过程分三步: 配置 the Hibernate SessionFactory. 继承 Hibern ...

  10. springboot-数据访问之jdbc

    官网的starthttps://docs.spring.io/spring-boot/docs/current/reference/htmlsingle/#using-boot-starter 111 ...