Spring Boot `@Service` 互相调用全攻略:`@Autowired` vs `@Resource`

Spring Boot @Service 互相调用全攻略:@Autowired vs @Resource

在日常写 Spring Boot 项目的时候,经常会遇到一个问题:多个 @Service 之间需要互相调用 ,到底该怎么写才优雅?用 @Autowired?用 @Resource?循环依赖怎么办?

本文就带你一口气整清楚,并配合 Demo 来对比,最后还会总结最佳实践。


1. 基础:@Service 是什么?

在 Spring 里,@Service 其实就是一个 特殊的 Bean 。它被 Spring 容器管理,创建、销毁、注入都由 Spring 来完成。

所以,不管你用 @Autowired 还是 @Resource,本质上都是 依赖注入(Dependency Injection)


2. @Autowired:Spring 派来的助手

2.1 特点

  • 默认按照 类型(byType)注入;
  • 如果有多个同类型的 Bean,会报错,需要用 @Qualifier 指定;
  • 支持构造器、字段、Setter 注入;
  • 可以加 required = false,让依赖变成可选。

2.2 Demo

java 复制代码
@Service
public class UserService {
    public String getUserName(Long id) {
        return "User-" + id;
    }
}

@Service
public class OrderService {
    private final UserService userService;

    // 构造器注入(推荐 ✅)
    @Autowired
    public OrderService(UserService userService) {
        this.userService = userService;
    }

    public void createOrder(Long userId) {
        System.out.println("订单用户: " + userService.getUserName(userId));
    }
}

3. @Resource:JDK 官方背书

3.1 特点

  • 来自 JSR-250 标准,算是"官方背书";
  • 默认按照 名称(byName)注入,找不到时再按照类型;
  • 常用在字段 / Setter 注入;
  • 不支持 required = false

3.2 Demo

java 复制代码
@Service
public class UserService {
    public String getUserName(Long id) {
        return "User-" + id;
    }
}

@Service
public class OrderService {
    @Resource
    private UserService userService; // 按字段名 userService 找 Bean

    public void createOrder(Long userId) {
        System.out.println("订单用户: " + userService.getUserName(userId));
    }
}

3.3 多实现类场景

java 复制代码
@Service("vipUserService")
public class VipUserService extends UserService {
    @Override
    public String getUserName(Long id) {
        return "VIP-" + id;
    }
}

@Service
public class OrderService {
    @Resource(name = "vipUserService")
    private UserService userService;

    public void createOrder(Long userId) {
        System.out.println("订单用户: " + userService.getUserName(userId));
    }
}

4. 循环依赖问题

有时候,你写着写着,就会掉进一个坑:两个 Service 互相依赖

java 复制代码
@Service
public class AService {
    @Autowired
    private BService bService;

    public void a() {
        System.out.println("A 调用");
        bService.b();
    }
}

@Service
public class BService {
    @Autowired
    private AService aService;

    public void b() {
        System.out.println("B 调用");
        aService.a();
    }
}

结果:启动失败,提示循环依赖。

4.1 循环调用的可视化

AService BService 调用 b() 调用 a() 再次调用 b() 再次调用 a() 无限递归,最终 StackOverflow 或启动失败 AService BService

4.2 解决方法

  • 重构代码 (最佳 ✅):抽出公共逻辑放到 CService,避免直接互调。
  • 延迟注入 (权宜之计):在其中一个依赖上加 @Lazy
java 复制代码
@Service
public class BService {
    private final AService aService;

    public BService(@Lazy AService aService) {
        this.aService = aService;
    }

    public void b() {
        System.out.println("B 调用");
        aService.a();
    }
}

5. 总结:到底用哪个?

  • 单实现类场景

    @Autowired@Resource 都行,推荐 构造器 + @Autowired

  • 多实现类场景
    @Resource(name="xxx") 更直观;
    @Autowired + @Qualifier("xxx") 也可以。

  • 循环依赖

    优先考虑 重构 ;不得已时用 @Lazy


6. 最佳实践建议

  1. 优先构造器注入,少用字段注入

    字段注入虽然写起来爽,但对测试和维护都不友好。

  2. @Autowired vs @Resource 没有绝对优劣

    • 倾向 Spring → 用 @Autowired
    • 倾向标准化 → 用 @Resource
  3. 不要用循环依赖当"快捷方式"

    那通常说明设计有问题。


@Autowired@Resource 之间纠结,远不如搞清楚你的代码结构更重要。

真正能写好 Service 之间调用的,不是靠注解,而是靠 架构设计


实用小工具

App Store 截图生成器应用图标生成器在线图片压缩Chrome插件-强制开启复制-护眼模式-网页乱码设置编码
乖猫记账,AI智能分类的最佳聊天记账App。
Elasticsearch可视化客户端工具

相关推荐
秋月的私语1 小时前
如何快速将当前的c#工程发布成单文件
android·java·c#
天***88961 小时前
使用python写一个应用程序要求实现微软常用vc++功能排查与安装功能
java
代码充电宝2 小时前
LeetCode 算法题【简单】283. 移动零
java·算法·leetcode·职场和发展
caibixyy4 小时前
Spring Boot 整合 Redisson 实现分布式锁:实战指南
spring boot·分布式·后端
码事漫谈4 小时前
C++编程陷阱:悬空引用检测方法与防范指南
后端
码事漫谈4 小时前
缓存友好的数据结构设计:提升性能的关键技巧
后端
ccccczy_5 小时前
Spring Security 深度解读:JWT 无状态认证与权限控制实现细节
java·spring security·jwt·authentication·authorization·securityfilterchain·onceperrequestfilter
Lin_Aries_04215 小时前
容器化 Tomcat 应用程序
java·linux·运维·docker·容器·tomcat
sheji34165 小时前
【开题答辩全过程】以 springboot高校社团管理系统的设计与实现为例,包含答辩的问题和答案
java·spring boot·后端
Terio_my5 小时前
Spring Boot 集成 Redis 缓存解决方案
spring boot·redis·缓存