Spring Boot 分层架构详解:Controller、Service、Mapper...

在构建企业级后端应用时,良好的分层架构 不仅能提升代码的可读性和可维护性,也能帮助团队实现职责清晰、功能复用的目标。本文将详细介绍

Spring Boot 项目中常见的分层结构,重点讲解 ControllerServiceMapper 各层的职责、使用方式与最佳实践。


📚 一、为什么要分层?

我们在开发 Spring Boot 项目时,并不是所有代码堆在一个类里,而是根据功能职责划分为不同的"层",以实现:

  • 职责清晰
  • 降低耦合
  • 易于测试
  • 易于扩展和维护

这就是典型的 MVC(Model-View-Controller)+ Service + DAO 架构。


🧱 二、Spring Boot 项目的典型分层结构

层级 英文名称 主要职责 是否写 SQL 是否处理业务 是否响应请求
表现层 Controller 接收前端 HTTP 请求、处理参数、调用业务逻辑、返回响应结果 ❌(只转发)
业务逻辑层 Service 封装业务逻辑、事务控制、调用数据访问接口、组织业务流程
数据访问层 Mapper / DAO 定义数据库操作,配合 XML 或注解执行 SQL
数据模型层 Entity / Model 封装数据库表结构对应的数据对象,进行前后端数据传输
配置层 Config 配置类,如数据源、拦截器、跨域处理、安全策略等
启动器层 Application 应用程序启动入口,加载配置并启动 Spring Boot 环境

📁 三、目录结构推荐

一个标准的 Spring Boot + MyBatis 项目结构如下:

复制代码
src/
 └── main/
     ├── java/com/example/demo/
     │   ├── controller/     // 接收请求
     │   ├── service/        // 编写业务逻辑
     │   ├── mapper/         // 与数据库交互
     │   ├── model/          // 数据对象(Entity)
     │   ├── config/         // 配置类
     │   └── DemoApplication.java
     └── resources/
         ├── mapper/         // MyBatis XML 文件
         └── application.yml // 配置文件

🧩 四、各层职责详解

✅ 1. Controller ------ 表现层

  • 接收 HTTP 请求(@GetMapping/@PostMapping
  • 处理请求参数(@RequestParam/@RequestBody
  • 调用 Service 进行业务处理
  • 返回响应结果(ResponseEntity/JSON

示例:

java 复制代码
@RestController
@RequestMapping("/users")
public class UserController {
    @Autowired
    private UserService userService;

    @GetMapping("/{id}")
    public User getById(@PathVariable Long id) {
        return userService.getUserById(id);
    }
}

✅ 2. Service ------ 业务逻辑层

  • 编写业务逻辑(如注册、下单、积分计算等)
  • 事务控制(使用 @Transactional
  • 调用多个 Mapper 或外部服务
  • 拆分复杂业务流程,提升可复用性

示例:

java 复制代码
@Service
public class UserService {
    @Autowired
    private UserMapper userMapper;

    public User getUserById(Long id) {
        return userMapper.selectById(id);
    }
}

✅ 3. Mapper ------ 数据访问层(DAO)

  • 编写 SQL 语句(使用 MyBatis 注解或 XML)
  • 与数据库进行交互(增删改查)
  • 不处理业务逻辑,仅作为数据库操作的封装器

示例(注解方式):

java 复制代码
@Mapper
public interface UserMapper {
    @Select("SELECT * FROM user WHERE id = #{id}")
    User selectById(Long id);
}

示例(XML 方式):

xml 复制代码
<mapper namespace="com.example.demo.mapper.UserMapper">
    <select id="selectById" resultType="User">
        SELECT * FROM user WHERE id = #{id}
    </select>
</mapper>

✅ 4. Model ------ 数据模型层

  • 定义实体类,用于封装数据库字段
  • 一般对应数据库中的一张表
  • 用作 Controller 与 Service 之间传递的数据对象

示例:

java 复制代码
@Data
public class User {
    private Long id;
    private String username;
    private String email;
}

💡 五、常见疑问解答

❓1. Controller 可以代替 Service 吗?

从技术上讲:可以写在一起。

从架构上讲:绝对不推荐!

👉 原因:

  • Controller 是面向接口的,只管"接收"与"响应"
  • Service 专注处理业务,解耦逻辑,有利于测试、复用、事务控制

❓2. Controller 可以代替 Mapper 或 XML 吗?

完全不能。

  • Controller 不应该操作数据库;
  • Mapper/XML 专门用来写 SQL;
  • 控制层不应承担数据库逻辑。

✅ 六、总结

Controller 管接口,Service 管逻辑,Mapper 管数据库,Model 管数据结构。


相关推荐
海上生明月丿11 小时前
微服务02
微服务·架构
麦兜*11 小时前
MongoDB 常见错误解决方案:从连接失败到主从同步问题
java·数据库·spring boot·redis·mongodb·容器
RestCloud11 小时前
iPaaS 与传统 ESB 的区别,企业该如何选择?
前端·架构
计算机学姐12 小时前
基于Python的旅游数据分析可视化系统【2026最新】
vue.js·后端·python·数据分析·django·flask·旅游
百度智能云12 小时前
MySQL内核革新:智能拦截全表扫描,百度智能云守护数据库性能与安全
架构
LQ深蹲不写BUG12 小时前
微服务事务管理利器:Seata 核心原理与实践指南
微服务·云原生·架构
该用户已不存在12 小时前
你没有听说过的7个Windows开发必备工具
前端·windows·后端
失散1313 小时前
分布式专题——5 大厂Redis高并发缓存架构实战与性能优化
java·redis·分布式·缓存·架构
David爱编程13 小时前
深入 Java synchronized 底层:字节码解析与 MonitorEnter 原理全揭秘
java·后端
KimLiu13 小时前
LCODER之Python:使用Django搭建服务端
后端·python·django