当前位置: 首页 > news >正文

springboot入门-业务逻辑核心service层

在 Spring Boot 中,Service 层是业务逻辑的核心,负责协调数据访问层(Repository 或 Mapper)和控制器层(Controller),处理业务规则、事务管理以及数据转换。以下是 Service 层的详细说明、常用注解及依赖注入方式:


1. Service 层的核心作用

  1. 业务逻辑处理:封装复杂的业务规则(如订单创建、用户权限校验)。
  2. 事务管理:通过 @Transactional 注解确保数据库操作的原子性。
  3. 异常处理:统一捕获并处理数据访问层的异常(如 DataAccessException)。
  4. DTO 转换:将实体类(Entity)转换为数据传输对象(DTO),避免暴露敏感字段。

2. Service 层的常用注解

(1) @Service
  • 作用
    将类标记为 Spring 管理的业务逻辑组件,自动注册为 Bean,供其他层(如 Controller)注入使用。
  • 位置:类声明上方。
  • 示例
    @Service
    public class UserService {// 业务方法...
    }
    
(2) @Transactional
  • 作用
    声明方法或类需要 事务管理。如果方法抛出异常(默认 RuntimeException),事务将回滚。
  • 关键参数
    参数作用示例
    rollbackFor指定触发回滚的异常类型@Transactional(rollbackFor = Exception.class)
    propagation事务传播行为(如 REQUIREDREQUIRES_NEW@Transactional(propagation = Propagation.REQUIRED)
    isolation事务隔离级别(如 READ_COMMITTED@Transactional(isolation = Isolation.READ_COMMITTED)
  • 示例
    @Service
    public class OrderService {@Transactional  // 方法级事务public void createOrder(Order order) {// 业务逻辑...}
    }
    

3. 如何将 Repository/Mapper 注入到 Service?

在 Spring Boot 中,通过 依赖注入(Dependency Injection, DI) 将 Repository 或 Mapper 注入到 Service。以下是两种常用方式:

(1) 构造器注入(推荐)
  • 原理:通过构造函数传递依赖,Spring 自动注入 Bean。
  • 优点
    • 明确依赖关系,避免 NullPointerException
    • 支持不可变(final)字段,线程安全。
  • 示例
    @Service
    public class UserService {private final UserRepository userRepository;// 构造器注入(Spring 4.3+ 可省略 @Autowired)public UserService(UserRepository userRepository) {this.userRepository = userRepository;}public User getUserById(Long id) {return userRepository.findById(id).orElse(null);}
    }
    
(2) @Autowired 注解注入
  • 原理:通过字段或 Setter 方法自动注入依赖。
  • 缺点
    • 字段注入可能导致循环依赖问题。
    • Setter 注入不够直观。
  • 示例
    @Service
    public class UserService {@Autowired  // 字段注入(不推荐)private UserRepository userRepository;// 或 Setter 注入@Autowiredpublic void setUserRepository(UserRepository userRepository) {this.userRepository = userRepository;}
    }
    

4. 完整示例:Service 层工作流程

(1) 实体类(Model)
@Entity
@Table(name = "users")
public class User {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;private String name;private String email;// Getter & Setter
}
(2) Repository 接口(JPA)
public interface UserRepository extends JpaRepository<User, Long> {User findByEmail(String email);
}
(3) Service 类
@Service
public class UserService {private final UserRepository userRepository;public UserService(UserRepository userRepository) {this.userRepository = userRepository;}@Transactionalpublic User createUser(String name, String email) {User user = new User();user.setName(name);user.setEmail(email);return userRepository.save(user);  // 调用 Repository 保存数据}public User getUserByEmail(String email) {return userRepository.findByEmail(email);}
}
(4) Controller 类
@RestController
@RequestMapping("/api/users")
public class UserController {private final UserService userService;public UserController(UserService userService) {this.userService = userService;}@PostMappingpublic User createUser(@RequestBody UserRequest request) {return userService.createUser(request.getName(), request.getEmail());}@GetMapping("/{email}")public User getUser(@PathVariable String email) {return userService.getUserByEmail(email);}
}

5. 常见问题解答

(1) 为什么推荐构造器注入?
  • 不可变性:字段标记为 final,避免中途修改。
  • 依赖明确:一眼看出 Service 需要哪些依赖。
  • 避免循环依赖:构造器注入在启动时就能发现循环依赖问题。
(2) @Transactional 应该加在 Service 还是 Repository?
  • 推荐加在 Service 层
    Service 方法可能涉及多个 Repository 操作,需统一事务管理(如用户注册时同时操作 User 表和 Account 表)。
(3) 如何处理多个数据源或 Mapper(MyBatis)?
  • MyBatis 的 Mapper 注入
    使用 @MapperScan 扫描 Mapper 接口,然后直接注入:
    @Service
    public class OrderService {private final OrderMapper orderMapper;public OrderService(OrderMapper orderMapper) {this.orderMapper = orderMapper;}
    }
    

总结

组件职责核心注解依赖注入方式
Service处理业务逻辑、事务管理@Service, @Transactional构造器注入(推荐)
Repository数据访问操作(JPA)无(继承 JpaRepository自动注入到 Service
Mapper数据访问操作(MyBatis)@Mapper(MyBatis)自动注入到 Service

通过合理使用 @Service@Transactional,结合构造器注入,可以构建清晰、可维护的 Service 层,高效管理业务逻辑和数据库事务。

相关文章:

  • Python-MCPServerStdio开发
  • Unity之基于MVC的UI框架-含案例
  • Linux——动静态库
  • 解决conda虚拟环境安装包却依旧安装到base环境下
  • 模板引擎语法-算术运算
  • springboot集成MyBatis Generator快速开发
  • 河南联通光猫超级管理员账号设置
  • 利用 functools.lru_cache 优化递归算法
  • Spark 极速回顾
  • ollama运行huggingface的gguf模型(mac)
  • 【Python】使用uv管理python虚拟环境
  • 数据结构之单链表C语言
  • React-Redux
  • 4.26-count部分的渲染
  • 基于STM32定时器中断讲解(HAL库)
  • 聊聊Spring AI Alibaba的YuQueDocumentReader
  • Rule.issuer(通过父路径配置loader处理器)
  • 启动你的RocketMQ之旅(五)-Broker详细——消息传输
  • 学习Spire.Office for Java版本的科学实践
  • 硬件须知的基本问题2
  • 阿斯利康中国区一季度收入增5%,或面临最高800万美元新罚单
  • 五一假期上海地铁部分线路将延时运营,这些调整请查收
  • 【社论】人工智能,年轻的事业
  • 西藏阿里地区日土县连发两次地震,分别为4.8级和3.8级
  • 阿里千问3系列发布并开源:称成本大幅下降,性能超越DeepSeek-R1
  • TAE联手加州大学开发出新型核聚变装置:功率提升百倍,成本减半