Spring Boot 实战(三):Service 分层 + 统一返回 + 异常处理(工程级写法)

tips: 先学习 ioc和di

Spring 核心思想解析:IoC 与 DI 一文讲透(从入门到工程理解)

一、前言

在前两篇中,我们已经完成:

复制代码
Controller 接口开发
DTO 参数接收
@Valid 参数校验

但当前代码仍然存在明显问题:

复制代码
Controller 承担了业务逻辑 ❌
返回结构不统一 ❌
没有异常处理 ❌
没有日志 ❌

👉 本篇将完成一次工程级升级


二、目标

我们将实现:

Controller → Service 分层 ✔

统一返回结构(Result)✔

统一异常处理 ✔

日志输出 ✔

三、统一返回结构(Result)

创建 Result 类

java 复制代码
package org.example.arkbackend.common;

import lombok.Data;

@Data
public class Result<T> {

    private int code;
    private String message;
    private T data;

    public static <T> Result<T> success(T data) {
        Result<T> r = new Result<>();
        r.setCode(0);
        r.setMessage("成功");
        r.setData(data);
        return r;
    }

    public static <T> Result<T> fail(String message) {
        Result<T> r = new Result<>();
        r.setCode(1);
        r.setMessage(message);
        return r;
    }
}

👉 返回统一格式:

java 复制代码
{
  "code": 0,
  "message": "成功",
  "data": {}
}

四、Service 层设计

1️⃣ 接口

java 复制代码
package org.example.arkbackend.service;

import org.example.arkbackend.dto.UserRegisterDTO;

public interface UserService {

    String register(UserRegisterDTO dto);

    String getUserById(Long id);

    void deleteUser(Long id);
}

2️⃣ 实现类

java 复制代码
package org.example.arkbackend.service.impl;

import lombok.extern.slf4j.Slf4j;
import org.example.arkbackend.dto.UserRegisterDTO;
import org.example.arkbackend.service.UserService;
import org.springframework.stereotype.Service;

@Slf4j
@Service
public class UserServiceImpl implements UserService {

    @Override
    public String register(UserRegisterDTO dto) {

        log.info("开始注册用户:{}", dto.getUsername());

        // 模拟业务逻辑
        if ("admin".equals(dto.getUsername())) {
            throw new RuntimeException("用户名已存在");
        }

        return "注册成功:" + dto.getUsername();
    }

    @Override
    public String getUserById(Long id) {
        log.info("查询用户ID:{}", id);
        return "用户ID:" + id;
    }

    @Override
    public void deleteUser(Long id) {
        log.info("删除用户ID:{}", id);
    }
}

五、Controller 重构

java 复制代码
package org.example.arkbackend.controller;

import jakarta.validation.Valid;
import org.example.arkbackend.common.Result;
import org.example.arkbackend.dto.UserRegisterDTO;
import org.example.arkbackend.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;

@RestController
@RequestMapping("/user")
public class UserController {

    @Autowired
    private UserService userService;

    @PostMapping("/register")
    public Result<String> register(@Valid @RequestBody UserRegisterDTO dto) {
        return Result.success(userService.register(dto));
    }

    @GetMapping("/{id}")
    public Result<String> getUser(@PathVariable Long id) {
        return Result.success(userService.getUserById(id));
    }

    @DeleteMapping("/{id}")
    public Result<Void> delete(@PathVariable Long id) {
        userService.deleteUser(id);
        return Result.success(null);
    }
}

👉 Controller 现在只负责:

复制代码
接收参数 + 调用 Service + 返回结果

六、统一异常处理(非常重要)

创建全局异常处理类

java 复制代码
package org.example.arkbackend.exception;

import org.example.arkbackend.common.Result;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;

@RestControllerAdvice
public class GlobalExceptionHandler {

    @ExceptionHandler(Exception.class)
    public Result<?> handleException(Exception e) {
        return Result.fail(e.getMessage());
    }
}

👉 当异常发生:

java 复制代码
{
  "code": 1,
  "message": "用户名已存在"
}

七、日志(Log)

使用 @Slf4j

复制代码
@Slf4j

使用方式

复制代码
log.info("注册用户:{}", username);
log.error("发生异常:{}", e.getMessage());

👉 优点:

复制代码
✔ 可追踪
✔ 可定位问题
✔ 可分析行为

八、完整执行流程

复制代码
请求进入 Controller
→ 参数校验(@Valid)
→ 调用 Service
→ Service 执行业务逻辑
→ 返回 Result
→ 异常统一处理

九、当前项目结构

复制代码
controller/
dto/
service/
service/impl/
common/
exception/

👉 已具备:

复制代码
企业级基础架构 ✔

十、一句话总结

复制代码
通过 Service 分层、统一返回结构和全局异常处理,可以将简单接口升级为具备工程规范的后端系统

下一篇:

Spring Boot 实战(四):MySQL + MyBatis 接入,打通用户注册最小闭环

复制代码
数据库(MySQL) + 持久层(Mapper / JPA)

👉 实现真实数据存储

相关推荐
云烟成雨TD1 小时前
Spring AI Alibaba 1.x 系列【20】MessagesAgentHook 、MessagesModelHook 相关实现类
java·人工智能·spring
霸道流氓气质1 小时前
SpringBoot中集成LangChain4j实现集成阿里百炼平台进行AI对话记忆功能和对话隔离功能
java·人工智能·spring boot·langchain4j
XS0301062 小时前
Java 基础笔记(二)
java·笔记·python
papaofdoudou2 小时前
AMD-V 嵌套分页白皮书翻译
java·linux·服务器
海寻山2 小时前
Java 泛型 (Generic) 入门到精通:语法 + 原理 + 实战 + 避坑
java·windows·python
艾莉丝努力练剑2 小时前
【Linux线程】Linux系统多线程(七):<线程同步与互斥>线程同步(下)
java·linux·运维·服务器·c++·学习·操作系统
云烟成雨TD2 小时前
Spring AI Alibaba 1.x 系列【15】工具执行拦截器(ToolInterceptor)
java·人工智能·spring
ch.ju2 小时前
Java程序设计(第3版)第二章——逻辑运算符
java
喜欢流萤吖~2 小时前
SpringBoot 异步处理与线程池实战
java·开发语言