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)

👉 实现真实数据存储

相关推荐
朦胧之8 小时前
AI 编程-老项目改造篇
java·前端·后端
程序猿大帅13 小时前
别再只当调包侠了:用 Spring AI 落地 Function Calling,我被大模型硬生生砸出了三个大坑
java
程序员晓琪14 小时前
约定大于配置:基于 Java 包名自动生成 API 版本路由的最佳实践
java·spring boot·后端
Flittly14 小时前
【AgentScope Java新手村系列】(11)中断与恢复
java·spring boot·spring
众少成多积小致巨14 小时前
JNI (Java Native Interface) 技术手册中文参考指南
android·java·c++
东坡白菜15 小时前
破局全栈:前端开发的Java入门实战记录—JPA(2)
java·后端
SimonKing21 小时前
艹,维护AI写的代码,我心态崩了......
java·后端·程序员
用户2986985301421 小时前
Java Word 文档样式进阶:段落与文本背景色设置完全指南
java·后端
小bo波2 天前
从"任意文件复制"深挖Java I/O:字符流与字节流的本质抉择
java·nio·io流·后端开发·文件复制
用户3521802454752 天前
🎆从 Prompt 到 Skill:让 Spring AI Agent 学会"装新技能"
人工智能·spring boot·ai编程