设计中 横向切割 和 纵向切割

本贴最后更新于 1807 天前,其中的信息可能已经时异事殊

自己学习一段时间 java 代码了,发现现在后端的模式大多都是三层架构,自顶向下的横向切割,而且基本都是 controller、service、dao 这种结构。

其实很多时候会发现一个问题,在某些小应用中,对于 service 来说,都是直接的一句话返回而已,但是却依旧贯彻 三层 分层,即使只有一句话都固执的去使用。为什么不舍得舍去呢?只用两层难道不行嘛?在某些实践中,我就舍掉了 service,减少了很多类不说,思路都会清晰横多,因为不用去维护更多的数据和逻辑了。

而且除了横向切割以外,为什么不能够纵向切割呢?例如以前我们的结构是这样的

+ src
|--- main
  |--- java
    |--- controller
      |--- UserController
    |--- service
      |--- UserService
    |--- entity
      |--- User
    |--- filter
      |--- UserFilter
    |--- dao
      |--- UserDao

那我们可以不可以这样恩

|--- main
  |--- java
    |--- user
      |--- UserController
      |--- UserService (个人觉得可以省掉)
      |--- UserDao
    |--- entity
      |--- User

这样的话个人觉得可以有几个好处:

  1. 专注于某个模块的实现
  2. 全栈工程师更加方便的 “一捅到底”
  3. 问题定位十分方便
  4. 将功能分模块细化

这样的设计我觉得也很棒啊,但是后端却很少见人实践,后面的项目就打算自己实践一下。

  • 设计
    111 引用 • 796 回帖 • 1 关注
  • MVC
    7 引用 • 119 回帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...
  • 88250

    这是包或者说是组件的设计原则决定的,归根结底就是

    • 高内聚、低耦合
    • 重用

    举个例子,DAO 必须要具有高度的可重用性,所以 xxxDao 都放在一个包里可以提升这个包的内聚,降低对外耦合,从何提升这个包的设计质量。如果分散到多个包里,那其他代码对 Dao 的依赖会变得错综复杂,从而打破每个包的内聚。

    另外,大部分业务逻辑实现都是通过“事务脚本”(《企业应用架构的模式》PoEAA)完成,通过 Service 层进行数据库事务封装,省掉 Srv 也不是不行,因为可以直接在 Ctl 里封装事务,具体看整体业务复杂度而言。一般来说为了保持整体设计的一致性,会保留 Srv 层,因为一开始可能业务简单,Srv 里面每个方法就一两句调用 Dao 的,但随着业务复杂起来,Srv 的实现就会丰富起来,这时也需要一定的重用,所以 Srv 层会从设计一开始就保留。

    另外,“过早的优化是万恶之源”,无论这个优化是指的简化还是复杂化,折中方案是设计上永恒的追求。

    1 回复
  • xflash

    这让我想起了好多年前大家都热衷于讨论领域模型,什么是贫血、充血、涨血等等。其实没什么意义,好的设计是大家都能理解的设计,好的设计是打开 IDE 就能找到写代码地方的设计。对于一个普通 CRUD 业务的项目,要为其写一堆“架构”文档才能让开发人员理解,那么做设计的人非傻即坏。

  • lizhongyue248

    最后总结到了一点

    不能太简,因为可能会为后来留下许多难以完成的事。

    不能太复杂,因为可能前期会花一些无用功。

    所以折中最为理想。

    但是如果换一个思路,横向切割保留了包的内聚,那么纵向切割是不是保留了模块的内聚呢?那么也就是提高了这个模块的设计质量。特别现在前端也是模块化/组件化,那么这样其实对应过来反而会好许多。

    而且,如果这些模块的操作不是与实体类对应的呢?而是与功能进行对应的呢?来考虑这种模块结构

    |--- main
      |--- java
        |--- auth
          |--- AuthController 授权的处理,登录/注册/找回密码等
          |--- AuthDao 授权操作数据库
        |--- entity
          |--- User
          |--- Role
    
    

    AuthDao 不一定只操作 User 表,他也要操作 Role 表,但是他可以选择是否调用 User 、Role 的 Dao,或者就是直接对这两张表进行操作,那么模块之间其实并没有太多的耦合了,而都是各个模块之间完成自己的事情,并不需要去调来调去。

    这样的设计完全颠覆了以前的横向切割的设计,是不是可取性更高呢?

    1 回复
  • 88250

    这个时候 AuthDao 其实就是 AuthService。Dao 一般只操作单表,Service 组合操作 Dao。

  • someone45057

    我还在某些地方看到这两个名词叫:水平拆分跟垂直拆分
    其实细细想来项目的发展过程就是水平跟垂直之间的平衡

    1 回复
  • lizhongyue248

    是的,各有所长各有所短,怎么平衡还是看需求来 ~

请输入回帖内容 ...
lizhongyue248
一个天真的小孩儿......https://echocow.cn 贵阳

推荐标签 标签

  • jsDelivr

    jsDelivr 是一个开源的 CDN 服务,可为 npm 包、GitHub 仓库提供免费、快速并且可靠的全球 CDN 加速服务。

    5 引用 • 31 回帖 • 42 关注
  • DNSPod

    DNSPod 建立于 2006 年 3 月份,是一款免费智能 DNS 产品。 DNSPod 可以为同时有电信、网通、教育网服务器的网站提供智能的解析,让电信用户访问电信的服务器,网通的用户访问网通的服务器,教育网的用户访问教育网的服务器,达到互联互通的效果。

    6 引用 • 26 回帖 • 521 关注
  • GraphQL

    GraphQL 是一个用于 API 的查询语言,是一个使用基于类型系统来执行查询的服务端运行时(类型系统由你的数据定义)。GraphQL 并没有和任何特定数据库或者存储引擎绑定,而是依靠你现有的代码和数据支撑。

    4 引用 • 3 回帖 • 20 关注
  • gRpc
    10 引用 • 8 回帖 • 52 关注
  • 安全

    安全永远都不是一个小问题。

    189 引用 • 813 回帖 • 2 关注
  • Ruby

    Ruby 是一种开源的面向对象程序设计的服务器端脚本语言,在 20 世纪 90 年代中期由日本的松本行弘(まつもとゆきひろ/Yukihiro Matsumoto)设计并开发。在 Ruby 社区,松本也被称为马茨(Matz)。

    7 引用 • 31 回帖 • 177 关注
  • IBM

    IBM(国际商业机器公司)或万国商业机器公司,简称 IBM(International Business Machines Corporation),总公司在纽约州阿蒙克市。1911 年托马斯·沃森创立于美国,是全球最大的信息技术和业务解决方案公司,拥有全球雇员 30 多万人,业务遍及 160 多个国家和地区。

    16 引用 • 53 回帖 • 118 关注
  • CSS

    CSS(Cascading Style Sheet)“层叠样式表”是用于控制网页样式并允许将样式信息与网页内容分离的一种标记性语言。

    180 引用 • 447 回帖 • 2 关注
  • 支付宝

    支付宝是全球领先的独立第三方支付平台,致力于为广大用户提供安全快速的电子支付/网上支付/安全支付/手机支付体验,及转账收款/水电煤缴费/信用卡还款/AA 收款等生活服务应用。

    29 引用 • 347 回帖 • 1 关注
  • 阿里云

    阿里云是阿里巴巴集团旗下公司,是全球领先的云计算及人工智能科技公司。提供云服务器、云数据库、云安全等云计算服务,以及大数据、人工智能服务、精准定制基于场景的行业解决方案。

    89 引用 • 345 回帖 • 2 关注
  • 安装

    你若安好,便是晴天。

    128 引用 • 1184 回帖 • 1 关注
  • Python

    Python 是一种面向对象、直译式电脑编程语言,具有近二十年的发展历史,成熟且稳定。它包含了一组完善而且容易理解的标准库,能够轻松完成很多常见的任务。它的语法简捷和清晰,尽量使用无异义的英语单词,与其它大多数程序设计语言使用大括号不一样,它使用缩进来定义语句块。

    535 引用 • 672 回帖 • 1 关注
  • GitBook

    GitBook 使您的团队可以轻松编写和维护高质量的文档。 分享知识,提高团队的工作效率,让用户满意。

    3 引用 • 8 回帖
  • ActiveMQ

    ActiveMQ 是 Apache 旗下的一款开源消息总线系统,它完整实现了 JMS 规范,是一个企业级的消息中间件。

    19 引用 • 13 回帖 • 628 关注
  • WiFiDog

    WiFiDog 是一套开源的无线热点认证管理工具,主要功能包括:位置相关的内容递送;用户认证和授权;集中式网络监控。

    1 引用 • 7 回帖 • 547 关注
  • Postman

    Postman 是一款简单好用的 HTTP API 调试工具。

    4 引用 • 3 回帖 • 1 关注
  • RabbitMQ

    RabbitMQ 是一个开源的 AMQP 实现,服务器端用 Erlang 语言编写,支持多种语言客户端,如:Python、Ruby、.NET、Java、C、PHP、ActionScript 等。用于在分布式系统中存储转发消息,在易用性、扩展性、高可用性等方面表现不俗。

    49 引用 • 60 回帖 • 392 关注
  • 游戏

    沉迷游戏伤身,强撸灰飞烟灭。

    169 引用 • 799 回帖
  • HBase

    HBase 是一个分布式的、面向列的开源数据库,该技术来源于 Fay Chang 所撰写的 Google 论文 “Bigtable:一个结构化数据的分布式存储系统”。就像 Bigtable 利用了 Google 文件系统所提供的分布式数据存储一样,HBase 在 Hadoop 之上提供了类似于 Bigtable 的能力。

    17 引用 • 6 回帖 • 44 关注
  • React

    React 是 Facebook 开源的一个用于构建 UI 的 JavaScript 库。

    192 引用 • 291 回帖 • 440 关注
  • 阿里巴巴

    阿里巴巴网络技术有限公司(简称:阿里巴巴集团)是以曾担任英语教师的马云为首的 18 人,于 1999 年在中国杭州创立,他们相信互联网能够创造公平的竞争环境,让小企业通过创新与科技扩展业务,并在参与国内或全球市场竞争时处于更有利的位置。

    43 引用 • 221 回帖 • 242 关注
  • 负能量

    上帝为你关上了一扇门,然后就去睡觉了....努力不一定能成功,但不努力一定很轻松 (° ー °〃)

    85 引用 • 1201 回帖 • 454 关注
  • LeetCode

    LeetCode(力扣)是一个全球极客挚爱的高质量技术成长平台,想要学习和提升专业能力从这里开始,充足技术干货等你来啃,轻松拿下 Dream Offer!

    209 引用 • 72 回帖
  • 大疆创新

    深圳市大疆创新科技有限公司(DJI-Innovations,简称 DJI),成立于 2006 年,是全球领先的无人飞行器控制系统及无人机解决方案的研发和生产商,客户遍布全球 100 多个国家。通过持续的创新,大疆致力于为无人机工业、行业用户以及专业航拍应用提供性能最强、体验最佳的革命性智能飞控产品和解决方案。

    2 引用 • 14 回帖
  • Sandbox

    如果帖子标签含有 Sandbox ,则该帖子会被视为“测试帖”,主要用于测试社区功能,排查 bug 等,该标签下内容不定期进行清理。

    368 引用 • 1212 回帖 • 576 关注
  • Dubbo

    Dubbo 是一个分布式服务框架,致力于提供高性能和透明化的 RPC 远程服务调用方案,是 [阿里巴巴] SOA 服务化治理方案的核心框架,每天为 2,000+ 个服务提供 3,000,000,000+ 次访问量支持,并被广泛应用于阿里巴巴集团的各成员站点。

    60 引用 • 82 回帖 • 602 关注
  • Sym

    Sym 是一款用 Java 实现的现代化社区(论坛/BBS/社交网络/博客)系统平台。

    下一代的社区系统,为未来而构建

    523 引用 • 4581 回帖 • 692 关注