Skip to content

后端分层架构

核心问题: 代码越写越乱,怎么组织才能清晰易懂?

当项目从几十行代码扩展到数万行,从单人开发到多人协作,从简单CRUD到复杂业务逻辑时,代码组织方式直接决定了项目的生死。分层架构不是为了炫技或遵循教条,而是为了解决软件工程中的一个根本性矛盾:业务复杂度的自然增长人类认知能力的有限性之间的冲突。


1. 为什么需要分层?

1.1 问题的根源

初期版本(100行代码):

java
@PostMapping("/register")
public Result register(@RequestBody User user) {
    // 1. 检查用户名是否重复
    if (userRepository.findByUsername(user.getUsername()) != null) {
        return Result.error("用户名已存在");
    }
    // 2. 加密密码
    user.setPassword(encrypt(user.getPassword()));
    // 3. 保存用户
    userRepository.save(user);
    // 4. 发送欢迎邮件
    emailService.sendWelcome(user.getEmail());
    // 5. 记录日志
    log.info("User registered: {}", user.getUsername());
    return Result.success();
}

6个月后(500行代码):

  • 新增了手机号验证
  • 新增了实名认证
  • 新增了邀请奖励
  • 新增了风控检查
  • ...

现在这个方法有500行,每次修改都提心吊胆,因为:

  • 逻辑混在一起,改一处可能影响其他功能
  • 难以测试,每次测试都要模拟完整的HTTP请求
  • 新人看不懂,因为所有逻辑都堆在一起

问题的本质:代码没有"边界",所有职责都混在一起。

技术债的累积效应:

  • 高耦合:业务逻辑与数据访问、HTTP协议耦合,修改牵一发而动全身
  • 低内聚:一个方法承担了多个职责,违反单一职责原则
  • 难测试:无法独立测试业务逻辑,必须启动完整HTTP容器
  • 难复用:业务逻辑绑定在HTTP请求中,定时任务、消息队列无法复用
  • 认知负荷:开发者需要同时理解所有层次的细节,无法聚焦

1.2 分层的核心思想

分层架构就是给代码划清边界:

┌─────────────────────────────────────┐
│  接收请求 ← Controller              │  只负责"接单"
├─────────────────────────────────────┤
│  业务编排 ← Service                 │  只负责"做菜"
├─────────────────────────────────────┤
│  数据存取 ← Repository              │  只负责"取食材"
├─────────────────────────────────────┤
│  业务定义 ← Domain                  │  只负责"菜谱标准"
└─────────────────────────────────────┘

关键原则:

  • 每一层只做自己的事
  • 层与层之间通过明确的接口通信
  • 业务逻辑集中在 Service 和 Domain
  • 数据访问逻辑集中在 Repository

分层架构的工程价值:

  1. 降低认知负荷:开发者可以专注于当前层的职责,无需理解全局细节
  2. 提高可测试性:每层可以独立单元测试,Mock依赖即可
  3. 增强可维护性:需求变更时,定位修改范围明确,降低风险
  4. 促进代码复用:业务逻辑不依赖HTTP,可在定时任务、消息队列中复用
  5. 支持团队协作:不同开发者可以并行开发不同层,减少冲突
  6. 延长代码寿命:清晰的边界让代码更容易重构和演进

2. 四层架构详解

2.1 整体结构

分层架构的本质是关注点分离(Separation of Concerns)和依赖方向控制:

┌─────────────────────────────────────────────────────┐
│  前端请求                                            │
└────────────────────┬────────────────────────────────┘
                     │ HTTP Request

┌─────────────────────────────────────────────────────┐
│  Controller (控制器层)                               │
│  - 接收请求、参数校验                                 │
│  - DTO 转换                                          │
│  - 调用 Service                                      │
│  - 返回响应                                          │
└────────────────────┬────────────────────────────────┘
                     │ 业务调用

┌─────────────────────────────────────────────────────┐
│  Service (业务逻辑层)                                │
│  - 业务逻辑编排                                      │
│  - 事务管理                                          │
│  - 协调多个 Repository                               │
│  - 跨模块协调                                        │
└────────────────────┬────────────────────────────────┘
                     │ 数据访问

┌─────────────────────────────────────────────────────┐
│  Repository (数据访问层)                             │
│  - 数据库 CRUD                                       │
│  - 查询封装                                          │
│  - ORM 映射                                          │
└────────────────────┬────────────────────────────────┘
                     │ 领域对象

┌─────────────────────────────────────────────────────┐
│  Domain (领域模型层)                                 │
│  - 实体 (Entity)                                     │
│  - 值对象 (Value Object)                             │
│  - 业务规则                                          │
└─────────────────────────────────────────────────────┘

依赖方向:代码依赖必须指向更稳定、更抽象的方向

  • Controller 依赖 Service 接口(抽象)
  • Service 依赖 Repository 接口(抽象)
  • 所有层都依赖 Domain(业务核心,最稳定)
  • 不允许反向依赖(如 Repository 依赖 Service)
后端四层架构总览
点击各层查看详细说明
客户端 (Web / App)
↓ HTTP
Controller入口
接收请求、参数校验、调用 Service
Service业务核心
业务逻辑编排、事务管理、跨模块协调
Repository数据访问
数据持久化、查询封装、ORM 映射
Domain领域模型
实体定义、业务规则、值对象
↓ SQL
数据库 (MySQL / PostgreSQL)

2.2 Controller 层

职责:请求的"接待员"

  • 接收 HTTP 请求,解析参数
  • 参数校验(格式、必填等)
  • DTO 转换(Request → Param)
  • 调用 Service 执行业务
  • DTO 转换(Result → Response)
  • 返回 HTTP 响应

不该做的事:

  • 直接写业务逻辑
  • 直接操作数据库
  • 处理事务

设计哲学: Controller 是系统的"门面",承担适配器职责——将外部HTTP协议适配为内部业务调用。它不应该包含任何业务决策,因为业务决策是领域知识的体现,应该与传输协议解耦。

示例:

java
@RestController
@RequestMapping("/api/users")
public class UserController {

    private final UserService userService;

    @PostMapping
    public UserResponse createUser(
            @RequestBody @Valid UserRequest request) {
        
        // 1. Request DTO → Param DTO
        UserParam param = UserParam.builder()
                .username(request.getUsername())
                .password(encrypt(request.getPassword()))
                .email(request.getEmail())
                .build();

        // 2. 调用 Service
        User user = userService.createUser(param);

        // 3. Entity → Response DTO
        return UserResponse.from(user);
    }
}

关键点:

  • @Valid 自动校验参数
  • 用 DTO 隔离前后端数据结构
  • 只做"翻译"和"调度",不包含业务逻辑
Controller 层:请求的"接待员"
点击流程节点查看详情
客户端发起请求
POST /api/users/register
Content-Type: application/json
{ "username": "张三", "email": "zhangsan@example.com", "password": "123456" }
↓ 请求到达
Controller 接收并解析请求
@RestController
@RequestMapping("/api/users")
public class UserController {
    @PostMapping("/register")
    public ResponseEntity<UserDTO> register(
        @RequestBody @Valid UserRegisterRequest request) {
        UserDTO user = userService.register(request);
        return ResponseEntity.ok(user);
    }
}
↓ 参数校验 + 调用
参数校验(Controller 的职责之一)
public class UserRegisterRequest {
    @NotBlank(message = "用户名不能为空")
    @Size(min = 2, max = 20) private String username;
    @Email(message = "邮箱格式不正确") private String email;
    @Size(min = 6, message = "密码至少6位") private String password;
}
↓ 返回结果
Controller 封装响应返回
HTTP/1.1 200 OK
{ "code": 200, "message": "注册成功",
  "data": { "id": 10001, "username": "张三", "email": "zhangsan@example.com" } }
Controller 的核心职责
接收请求
映射 HTTP 请求到方法
参数校验
基础格式和必填校验
调用 Service
将请求转发给业务层
封装响应
统一响应格式返回

2.3 Service 层

职责:业务的"厨师"

  • 实现核心业务逻辑
  • 编排多个 Repository 的操作
  • 管理事务边界
  • 处理跨模块协调

不该做的事:

  • 直接写 SQL(交给 Repository)
  • 处理 HTTP 相关的事情
  • 返回数据库实体给 Controller

设计哲学: Service 层是业务逻辑的载体,应该保持纯粹性。它不依赖任何框架或传输协议,这样可以:

  • 独立于Web层进行单元测试
  • 在定时任务、消息队列消费者中复用
  • 避免技术栈变更影响业务逻辑

示例:

java
@Service
@RequiredArgsConstructor
public class UserService {

    private final UserRepository userRepository;
    private final EmailService emailService;

    @Transactional
    public User createUser(UserParam param) {
        // 1. 业务规则:检查用户名是否重复
        if (userRepository.existsByUsername(param.getUsername())) {
            throw new UserAlreadyExistsException();
        }

        // 2. 创建用户实体
        User user = new User();
        user.setUsername(param.getUsername());
        user.setPassword(param.getPassword());
        user.setEmail(param.getEmail());

        // 3. 保存到数据库
        userRepository.save(user);

        // 4. 发送欢迎邮件(跨模块协调)
        emailService.sendWelcomeEmail(user);

        return user;
    }
}

关键点:

  • 用Transactional保证事务一致性
  • 抛出业务异常,让Controller统一处理
  • 不依赖HTTP概念,可以复用
Service 层:业务逻辑的"指挥家"
选择业务场景,查看 Service 层如何编排逻辑
电商下单流程
用户下单涉及库存扣减、订单创建、支付记录,需保证事务一致性
1
参数校验与DTO转换
Controller
@PostMapping("/orders")
public ResponseEntity<OrderDTO> createOrder(
    @RequestBody @Valid CreateOrderRequest request) {
    OrderDTO order = orderService.createOrder(request);
    return ResponseEntity.ok(order);
}
2
业务逻辑编排(事务管理)
Service
@Transactional
public OrderDTO createOrder(CreateOrderRequest request) {
    inventoryService.checkAndDeduct(request.getSkuId(), request.getQuantity());
    Order order = new Order();
    order.setUserId(request.getUserId());
    order.setTotalAmount(calculateTotal(request));
    orderRepository.save(order);
    Payment payment = createPayment(order);
    paymentRepository.save(payment);
    return convertToDTO(order);
}
3
数据持久化
Repository
public interface OrderRepository extends JpaRepository<Order, Long> {
    // 基本 CRUD 已内置
}
Service 层设计原则
单一职责
一个 Service 只负责一块业务领域
UserService 只管用户,OrderService 只管订单
事务边界
在 Service 层声明式管理事务
@Transactional 放在 Service 方法上
避免循环依赖
Service 之间不要互相调用
A→B→A 会导致循环
DTO 转换
返回前转换为 DTO,不暴露实体
return new UserDTO(user)

2.4 Repository 层

职责:数据的"仓管员"

  • 封装所有数据访问逻辑
  • 执行CRUD操作
  • 处理ORM映射
  • 封装查询条件

不该做的事:

  • 写业务逻辑
  • 处理事务(Service层管理)
  • 依赖上层模块

设计哲学: Repository 是数据访问的抽象层,它隐藏了底层数据库的细节。这种抽象的价值在于:

  • 切换数据库时只需修改Repository实现,业务逻辑无需变动
  • 便于Mock进行单元测试
  • 查询逻辑集中管理,避免重复代码

示例:

java
@Repository
public interface UserRepository extends JpaRepository<User, Long> {

    // Spring Data JPA 自动实现
    Optional<User> findByUsername(String username);
    boolean existsByUsername(String username);

    // 自定义复杂查询
    @Query("SELECT u FROM User u WHERE u.email = :email AND u.deleted = false")
    Optional<User> findActiveByEmail(@Param("email") String email);
}

关键点:

  • Repository是接口,不包含业务逻辑
  • 用方法名表达查询意图
  • 可以用Query自定义复杂查询
Repository 层:数据的"仓库管理员"
Repository 封装数据访问逻辑,让上层无需关心数据库细节
使用 Repository 封装数据访问清晰解耦
// Repository 接口定义
@Repository
public interface OrderRepository extends JpaRepository<Order, Long> {
    // ✅ 自动生成查询
    List<Order> findByUserIdAndDeletedFalse(Long userId);

    // ✅ 自定义 JPQL
    @Query("SELECT o FROM Order o WHERE o.createdAt BETWEEN :start AND :end")
    List<Order> findByDateRange(@Param("start") LocalDateTime start,
                                @Param("end") LocalDateTime end);
}

// Service 层(纯业务逻辑)
@Service
public class OrderService {
    @Autowired private OrderRepository orderRepository; // ✅ 依赖接口

    public List<OrderDTO> getUserOrders(Long userId) {
        List<Order> orders = orderRepository.findByUserIdAndDeletedFalse(userId);
        return orders.stream().map(OrderDTO::from).collect(Collectors.toList());
    }
}
这样做的好处
  • 关注点分离:Service 专注业务,Repository 专注数据
  • 可测试性高:单元测试可用 Mock 替代真实数据库
  • 代码复用:通用查询方法定义一次,到处复用
  • 切换成本低:换数据库只需改 Repository 实现
不同 Repository 实现方式对比
实现方式优点缺点适用场景
Spring Data JPA
主流方案
方法名自动推导、分页内置复杂查询性能一般快速开发、标准 CRUD
MyBatis / MyBatis-Plus
国内主流
SQL 完全可控、动态 SQL 强大需要手写 SQL复杂查询、性能敏感
Spring Data JDBC
轻量
简单轻量、启动快速无复杂映射微服务、简单聚合根

2.5 Domain 层

职责:业务的"菜谱标准"

  • 定义业务实体(Entity)
  • 定义值对象(Value Object)
  • 封装业务规则
  • 作为所有层的共同依赖

重要特性:

  • Domain层不依赖任何其他层
  • 所有层都依赖Domain层
  • 是分层架构的基础

设计哲学: Domain层是整个系统的业务核心,它表达了领域知识和业务规则。它的纯粹性至关重要:

  • 不依赖框架意味着业务逻辑不被技术栈绑架
  • 所有层都依赖它,保证了业务规则的统一性
  • 便于长期演进,技术栈可以替换,业务规则相对稳定

示例:

java
@Entity
public class User {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;

    @Column(unique = true, nullable = false)
    private String username;

    @Column(nullable = false)
    private String password;

    // ✅ 业务方法:封装业务规则
    public boolean isPasswordCorrect(String rawPassword) {
        return BCrypt.checkpw(rawPassword, this.password);
    }

    public void changePassword(String oldPassword, String newPassword) {
        if (!isPasswordCorrect(oldPassword)) {
            throw new IncorrectPasswordException();
        }
        this.password = BCrypt.hashpw(newPassword);
    }
}

关键点:

  • Entity 有唯一标识
  • 业务规则封装在 Domain 对象中
  • Domain 层是纯粹的业务逻辑,不依赖框架
Domain 层:领域模型设计
Domain 是业务概念的载体,所有层的依赖基础
贫血模型 (Anemic)传统做法
@Entity
public class Order {
    @Id private Long id;
    private BigDecimal totalAmount;
    private OrderStatus status;
    // 只有 getter/setter,没有业务逻辑
    public Long getId() { return id; }
    public void setStatus(OrderStatus s) { this.status = s; }
}
@Service
public class OrderService {
    public void cancelOrder(Long orderId) {
        Order order = orderRepository.findById(orderId).orElseThrow();
        // 贫血模型:业务逻辑散落在 Service 里
        if (order.getStatus() == OrderStatus.SHIPPED)
            throw new IllegalStateException("已发货不能取消");
        order.setStatus(OrderStatus.CANCELLED);
        orderRepository.save(order);
    }
}
贫血模型的问题
  • 违背面向对象:对象只有数据没有行为
  • 逻辑分散:同样的规则可能在多个 Service 重复
  • 难以维护:改一个规则要找所有用到的地方
充血模型 (Rich Domain)推荐做法
@Entity
public class Order {
    @Id private Long id;
    private BigDecimal totalAmount;
    private OrderStatus status;

    // 业务行为封装在实体里
    public void cancel() {
        if (this.status == OrderStatus.SHIPPED)
            throw new IllegalStateException("已发货不能取消");
        this.status = OrderStatus.CANCELLED;
        registerEvent(new OrderCancelledEvent(this.id));
    }

    public void pay(Payment payment) {
        if (this.status != OrderStatus.PENDING_PAYMENT)
            throw new IllegalStateException("状态不正确");
        this.status = OrderStatus.PAID;
    }
}
@Service
public class OrderService {
    @Transactional
    public void cancelOrder(Long orderId) {
        Order order = orderRepository.findById(orderId).orElseThrow();
        order.cancel(); // 调用领域对象的业务方法
        orderRepository.save(order);
    }
}
充血模型的优势
  • 符合面向对象:数据和行为封装在一起
  • 业务内聚:规则跟着对象走,改一处处处生效
  • 可测试:领域对象是纯内存对象,不需要数据库
  • 表达力强:order.cancel() 比 orderService.cancel(order) 更自然

3. DTO:层与层之间的"翻译官"

3.1 为什么需要 DTO?

问题:如果直接把数据库实体返回给前端:

java
// ❌ 错误:直接返回 Entity
@Entity
public class User {
    private Long id;
    private String username;
    private String password;        // 敏感信息!
    private Boolean isDeleted;      // 内部字段!
}

前端会收到不该暴露的字段,存在安全风险。

解决方案:用 DTO 做"翻译"

数据库 Entity → Service Param/Result → Controller Request/Response → 前端

3.2 DTO 的类型

类型用途示例
Request DTOController 接收参数UserCreateRequest
Response DTOController 返回数据UserResponse
Param DTOService 方法参数UserParam
Result DTOService 返回结果UserResult
Entity数据库映射User

关键原则: 每层使用自己的 DTO,不要直接传递 Entity,DTO 只包含必要的字段,这样可以避免暴露内部实现细节,保证各层的独立性。

DTO 流转:数据在不同层之间的转换
DTO(Data Transfer Object)是层与层之间传递数据的载体
Controller 层
// 接收 Request DTO
public ResponseEntity<UserDTO> createUser(
    @RequestBody @Valid UserCreateRequest request) { ... }
↓ 转换为 Service 需要的参数
Service 层
public UserDTO createUser(UserCreateParam param) {
    User user = param.toEntity();   // 转换为 Entity
    userRepository.save(user);
    return UserDTO.from(user);      // Entity → DTO
}
↓ 转换为 Repository 需要的 Entity
Repository 层
public interface UserRepository
    extends JpaRepository<User, Long> { }
↑ 返回 Entity,转换为 DTO
返回给客户端
{ "id": 10001, "username": "张三",
  "email": "zhangsan@example.com", "createdAt": "2024-01-15T10:30:00Z" }
不同层的 DTO 职责
层级DTO 类型职责示例
ControllerRequest / Response DTO定义 API 契约、参数校验UserCreateRequest
ServiceParam / Result DTO封装业务方法参数,解耦层间依赖UserCreateParam
RepositoryEntity / DO映射数据库表结构UserEntity

4. 依赖方向:分层架构的铁律

4.1 依赖倒置原则

错误的做法:

Controller → UserServiceImpl → UserDaoImpl → UserEntity

正确做法:

Controller → UserService(接口) → UserRepository(接口) → UserEntity

依赖方向:

正确的依赖方向是所有层都依赖更抽象、更稳定的层。具体来说,Controller 依赖 Service 接口,Service 依赖 Repository 接口,所有层都依赖 Domain 层,而 Domain 层不依赖任何其他层。这种依赖方向确保了业务逻辑的独立性和可测试性。

错误的做法包括 Service 直接依赖 Repository 实现类,Controller 直接操作数据库,或者 Domain 层依赖其他层,这些都会导致耦合度升高,降低系统的可维护性。

4.2 代码示例

java
// ✅ 正确:依赖接口
@Service
public class OrderService {
    private final OrderRepository orderRepository;  // 接口
    private final PaymentService paymentService;    // 接口
}

// ✅ 实现类通过 Spring 自动注入
@Repository
public class OrderRepositoryImpl implements OrderRepository {
    // 实现细节
}
依赖方向:分层架构的核心规则
理解依赖方向,才能真正掌握分层架构
外层(UI / 外部系统)
Controller
↓ 依赖
中层(应用层)
Service
↓ 依赖
内层(领域层)
Domain / Repository
核心原则:依赖倒置(DIP)

上层模块不应该依赖下层模块的具体实现,而应该依赖于抽象。

Controller → Service 接口
Controller 只依赖 Service 的接口,不依赖实现类
Service → Repository 接口
Service 只依赖 Repository 接口,不关心数据怎么存
所有层依赖 Domain
Domain 是核心,被所有上层依赖,但 Domain 不依赖任何层

5. 实战案例:电商订单系统

5.1 需求

创建订单:

  1. 用户选择商品
  2. 检查库存
  3. 计算金额
  4. 创建订单
  5. 扣减库存

5.2 代码实现

Domain 层:

java
@Entity
public class Order {
    @Id
    private Long id;
    private Long userId;
    private List<OrderItem> items;
    private Money totalAmount;
    private OrderStatus status;

    public void calculateTotal() {
        Money total = Money.zero();
        for (OrderItem item : items) {
            total = total.add(item.getSubTotal());
        }
        this.totalAmount = total;
    }

    public void cancel() {
        if (this.status != OrderStatus.PENDING_PAYMENT) {
            throw new IllegalStateException("只有待支付订单可以取消");
        }
        this.status = OrderStatus.CANCELLED;
    }
}

Repository 层:

java
@Repository
public interface OrderRepository extends JpaRepository<Order, Long> {
    List<Order> findByUserIdOrderByCreatedAtDesc(Long userId);
}

Service 层:

java
@Service
@RequiredArgsConstructor
public class OrderService {

    private final OrderRepository orderRepository;
    private final InventoryService inventoryService;

    @Transactional
    public OrderDTO createOrder(OrderParam param) {
        // 1. 验证商品并扣减库存
        for (OrderItemParam item : param.getItems()) {
            inventoryService.reserveStock(item.getProductId(), item.getQuantity());
        }

        // 2. 创建订单
        Order order = new Order();
        order.setUserId(param.getUserId());
        order.calculateTotal();

        // 3. 保存订单
        orderRepository.save(order);

        return OrderDTO.from(order);
    }
}

Controller 层:

java
@RestController
@RequestMapping("/api/orders")
public class OrderController {

    private final OrderService orderService;

    @PostMapping
    public OrderResponse createOrder(@RequestBody @Valid OrderRequest request) {
        OrderParam param = OrderParam.builder()
                .userId(request.getUserId())
                .items(request.getItems())
                .build();

        OrderDTO order = orderService.createOrder(param);

        return OrderResponse.from(order);
    }
}

6. 常见问题

6.1 Controller 可以写业务逻辑吗?

Controller 不应该写业务逻辑,它只负责接收请求和返回响应。业务逻辑应该封装在 Service 层,这样做的好处是代码可以被复用,例如定时任务或消息队列消费者可以直接调用 Service,而不需要通过 HTTP 请求。同时,业务逻辑集中在一个地方,更容易测试和维护,避免了逻辑分散导致的不一致问题。

6.2 什么是贫血模型和充血模型?

贫血模型是指实体类只包含属性和对应的 getter/setter 方法,不包含任何业务逻辑,所有的业务规则都放在 Service 层中实现。这种模型结构简单,易于理解,是大多数项目采用的方式。

充血模型是指实体类不仅包含属性,还包含与该实体相关的业务方法,将业务规则封装在实体内部。这种方式更符合面向对象的设计思想,让数据和行为在一起,提高了代码的内聚性。

建议根据团队的技术背景和项目复杂度选择合适的模型,但无论选择哪种,都应该保持一致性,并且 Domain 层至少应该包含基本的业务行为方法,而不是完全的空壳。

6.3 如何处理跨多个 Service 的事务?

当一个业务操作需要跨越多个 Service 时,应该在上层的 Service 中使用事务注解,在这个方法中依次调用多个下层的 Service。这样可以确保所有操作在同一个事务上下文中执行,要么全部成功要么全部失败,保证数据的一致性。需要注意的是,事务边界应该尽可能小,只包含必要的操作,避免长时间持有数据库锁影响并发性能。


7. 总结

层级职责关键词
Controller接收请求、参数校验、调用 Service、返回响应接待员
Service业务逻辑编排、事务管理、协调 Repository厨师
Repository数据访问、ORM 映射、查询封装仓管员
Domain实体定义、业务规则、值对象菜谱标准

���心原则:

  1. 每层只做自己的事
  2. 层与层之间通过接口通信
  3. 业务逻辑集中在 Service 和 Domain
  4. 数据访问逻辑集中在 Repository
  5. 用 DTO 隔离各层数据结构

8. 更多架构模式

本文介绍的是分层架构(Layered Architecture),这是最常见、最易上手的后端架构模式。但后端架构远不止这一种,根据业务场景不同,还有其他值得了解的架构模式:

8.1 其他常见架构模式

架构模式适用场景特点
单体架构小型项目、MVP所有功能在一个应用中,部署简单
微服务架构大型复杂系统拆分为多个独立服务,每个服务可独立部署
事件驱动架构高并发、异步处理通过事件触发处理流程,解耦度高
整洁架构复杂业务系统业务逻辑居中,依赖只能向内,框架在最外层
六边形架构需要多种外部适配通过端口和适配器隔离核心与外部系统
洋葱架构领域驱动设计同心圆分层,领域模型在最内层,基础设施在最外层

下面逐一展开介绍:

单体架构 (Monolithic)

所有功能打包在一个应用中,共享同一个数据库和进程。

┌──────────────────────────────┐
│         单体应用              │
│  ┌────┐ ┌────┐ ┌────┐       │
│  │用户│ │订单│ │支付│ ...    │
│  └──┬─┘ └──┬─┘ └──┬─┘       │
│     └──────┼──────┘          │
│         共享数据库            │
└──────────────────────────────┘
  • 优点: 开发简单、部署方便、本地调试容易
  • 缺点: 代码耦合度高,扩展困难,一个模块出问题可能拖垮整个系统
  • 适用: 早期创业项目、单团队开发、快速原型验证

微服务架构 (Microservices)

将系统拆分为多个独立服务,每个服务拥有自己的数据和业务逻辑,可独立部署和扩展。

┌────────┐  ┌────────┐  ┌────────┐
│用户服务 │  │订单服务 │  │支付服务 │
│  DB-1  │  │  DB-2  │  │  DB-3  │
└───┬────┘  └───┬────┘  └───┬────┘
    └───────────┼───────────┘
          API Gateway
  • 优点: 独立部署和扩展、技术栈灵活、故障隔离
  • 缺点: 服务间通信复杂、分布式数据一致性难、需要成熟的 DevOps 能力
  • 适用: 大型复杂系统、多团队协作、需要独立扩展的场景

事件驱动架构 (Event-Driven)

通过异步事件进行通信,生产者发出事件,消费者响应事件,组件之间高度解耦。

生产者 ──→ [事件总线/消息队列] ──→ 消费者A
                               ──→ 消费者B
                               ──→ 消费者C
  • 优点: 高度解耦、天然支持扩展、适合实时处理
  • 缺点: 调试困难、事件顺序和幂等性需要额外处理
  • 适用: 实时数据分析、IoT 系统、微服务间异步通信

整洁架构 (Clean Architecture)

Robert C. Martin 提出,将系统分为四个同心圆层,依赖只能从外向内指向:

┌─────────────────────────────────────┐
│  Frameworks & Drivers (框架和驱动)   │
│  ┌─────────────────────────────┐    │
│  │  Interface Adapters (适配器) │    │
│  │  ┌─────────────────────┐    │    │
│  │  │  Use Cases (用例)    │    │    │
│  │  │  ┌─────────────┐    │    │    │
│  │  │  │  Entities    │    │    │    │
│  │  │  │  (实体/领域)  │    │    │    │
│  │  │  └─────────────┘    │    │    │
│  │  └─────────────────────┘    │    │
│  └─────────────────────────────┘    │
└─────────────────────────────────────┘
         依赖方向: 外 → 内
  • 核心规则: 内层不知道外层的存在,业务逻辑完全独立于框架和数据库
  • 优点: 高可测试性、技术栈可替换、业务逻辑清晰
  • 缺点: 初期开发成本高、层间映射代码多、小项目容易过度设计
  • 适用: 复杂业务系统、需要长期维护的项目
整洁架构与分层架构对比
分层架构是整洁架构的基础,理解两者关系有助于构建更灵活的系统
Controller 层 接收请求、参数校验
Service 层 业务逻辑、事务管理
Repository 层 数据访问、ORM 映射
Domain 层 实体定义、业务规则
传统分层架构特点
  • 垂直依赖:上层直接依赖下层
  • 简单直观:结构清晰,易于理解
  • 适合中小型项目:快速开发,上手简单
  • 潜在问题:底层变更可能影响上层

六边形架构 (Hexagonal / Ports & Adapters)

通过"端口"定义核心业务的输入输出接口,通过"适配器"连接外部系统:

        ┌─────────────┐
  HTTP ──→ Port      │
  CLI  ──→ (入端口)   │  核心业务逻辑  │  (出端口) ──→ 数据库
  MQ   ──→           │               │  Port    ──→ 外部API
        └─────────────┘
  • 核心思想: 业务逻辑不依赖任何外部技术,外部系统通过适配器接入
  • 优点: 外部系统可随意替换、测试时用 Mock 适配器即可
  • 适用: 需要对接多种外部系统的场景

洋葱架构 (Onion Architecture)

与整洁架构类似,强调领域模型在最内层,基础设施在最外层,依赖只能向内:

┌──────────────────────────────┐
│  Infrastructure (基础设施)    │
│  ┌────────────────────────┐  │
│  │  Application Services  │  │
│  │  ┌──────────────────┐  │  │
│  │  │  Domain Services  │  │  │
│  │  │  ┌────────────┐   │  │  │
│  │  │  │Domain Model│   │  │  │
│  │  │  └────────────┘   │  │  │
│  │  └──────────────────┘  │  │
│  └────────────────────────┘  │
└──────────────────────────────┘
  • 核心思想: 领域模型是系统的核心,所有依赖都指向它
  • 与整洁架构的区别: 洋葱架构更强调领域服务层,整洁架构更强调用例层
  • 适用: 采用领域驱动设计(DDD)的项目

8.2 架构演进路线

这些架构不是互相替代的关系,而是逐步演进的:

text
传统分层架构 (N-Layered)
  │  问题: 层间耦合、难以替换外部依赖

六边形架构 (Ports & Adapters)
  │  改进: 用端口和适配器隔离外部系统

洋葱架构 (Onion)
  │  改进: 明确同心圆分层,领域模型居中

整洁架构 (Clean Architecture)
  │  改进: 统一依赖规则,明确四层职责

根据业务需要选择合适的架构

8.3 架构模式选择指南

text
用户量 < 1k, 代码量 < 5000 行

单体架构 + 简单分层

用户量 1k-100k, 需要多团队协作

分层架构 (本文介绍)

用户量 > 100k, 业务复杂度高

微服务架构 / 事件驱动架构

更细化的选择维度:

考虑因素简单分层整洁/六边形架构微服务
团队规模1-5 人5-20 人20+ 人
业务复杂度中高
部署频率高(独立部署)
技术栈多样性单一单一可多样
运维成本

8.4 推荐阅读

  • 单体架构: 查看本文的姐妹篇 backend-project-architecture.md,了解从脚本到单体的演进
  • 微服务架构: 查看 从单体到微服务的演进
  • 整洁架构: Robert C. Martin 的《Clean Architecture》— 提出依赖规则和四层同心圆模型的经典著作
  • 企业架构模式: Martin Fowler 的《Patterns of Enterprise Application Architecture》— 分层架构、领域逻辑组织的权威参考

8.5 如何选择?

记住这个原则: 架构服务于业务,不是为架构而架构

  • 小项目用简单架构,快速上线验证
  • 大项目再考虑复杂架构,避免过度设计
  • 团队熟悉度也很重要,选择大家都能理解的方案

9. 总结

层级职责关键词
Controller接收请求、参数校验、调用 Service、返回响应接待员
Service业务逻辑编排、事务管理、协调 Repository厨师
Repository数据访问、ORM 映射、查询封装仓管员
Domain实体定义、业务规则、值对象菜谱标准

核心原则:

分层架构的核心在于明确的职责划分和依赖方向控制。每一层只关注自己的职责,通过接口与相邻层通信,业务逻辑集中在 Service 和 Domain 层,数据访问逻辑集中在 Repository 层,各层之间通过 DTO 隔离数据结构,避免直接暴露内部实现。这样的设计让系统更易于理解、测试和维护,能够应对业务的持续演进。


参考资料

  1. Catalog of Patterns of Enterprise Application Architecture - Martin Fowler — Martin Fowler 的企业应用架构模式目录,分层架构的经典参考
  2. Backend Side Architecture Evolution (N-layered, DDD, Hexagon, Onion, Clean Architecture) — 从 N 层架构到整洁架构的演进历程,理解每种架构诞生的原因
  3. Complete Guide to Clean Architecture - GeeksforGeeks — 整洁架构完整指南,详解分层、依赖规则与关注点分离
  4. Understanding Hexagonal, Clean, Onion, and Traditional Layered Architectures: A Deep Dive — 六边形、整洁、洋葱与传统分层架构的深度对比
  5. Building Clean Architectures in Modern Backend Frameworks — 在现代后端框架中实践整洁架构的实战指南
  6. Backend Architecture Patterns: From Monoliths to Microservices — 从单体到微服务的后端架构模式全景概览
  7. MVC 三层架构案例详细讲解 — MVC 与三层架构的关系及实战案例,适合中文读者入门