正确使用@Resource

目录

  • [1 怎么使用`@Resource`?](#1 怎么使用@Resource?)
    • [1.0 实验环境](#1.0 实验环境)
    • [1.1 通过字段注入依赖](#1.1 通过字段注入依赖)
    • [1.2 bean property setter methods (setter方法)](#1.2 bean property setter methods (setter方法))
  • [2 打破岁月静好(`@Resource takes a name attribute`)](#2 打破岁月静好(@Resource takes a name attribute))
    • [2.1 结论](#2.1 结论)
    • [2.2 那我不指定呢?【结论:又能正常执行了】](#2.2 那我不指定呢?【结论:又能正常执行了】)
      • [2.2.1 default name](#2.2.1 default name)
  • [3 总结【@Resource是by-name来注入依赖的】](#3 总结【@Resource是by-name来注入依赖的】)

1 怎么使用@Resource

参考官方文档

1.0 实验环境

  • 项目结构:
  • 代码
java 复制代码
@Configuration
public class LearnResourceConfig {
    @Bean
    public MovieFinder movieFinder() {
        return new MovieFinder();
    }
}

public class MovieFinder {
    public void sayHello() {
        System.out.println(this.getClass().getSimpleName() + ", hello");
    }
}

1.1 通过字段注入依赖

  • SimpleMovieLister
java 复制代码
@Component
public class SimpleMovieLister {
    @Resource(name = "movieFinder")
    private MovieFinder movieFinder;

    public void sayHello() {
        movieFinder.sayHello();
        System.out.println(this.getClass().getSimpleName() + ", hello");
    }
}
  • Application
java 复制代码
@ComponentScan
public class Application {
    public static void main(String[] args) {
        AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(Application.class);
        Arrays.stream(applicationContext.getBeanDefinitionNames()).forEach(System.out::println);
        System.out.println("---------------------------------------------------------------");
        SimpleMovieLister simpleMovieLister = applicationContext.getBean(SimpleMovieLister.class);
        simpleMovieLister.sayHello();
    }
}

/*
org.springframework.context.event.internalEventListenerFactory
application
simpleMovieLister
learnResourceConfig
movieFinder
---------------------------------------------------------------
MovieFinder, hello
SimpleMovieLister, hello
*/

1.2 bean property setter methods (setter方法)

  • SimpleMovieLister
java 复制代码
@Component
public class SimpleMovieLister {
    private MovieFinder movieFinder;

    @Resource(name = "movieFinder")
    public void setMovieFinder(MovieFinder movieFinder) {
        this.movieFinder = movieFinder;
    }

    public void sayHello() {
        movieFinder.sayHello();
        System.out.println(this.getClass().getSimpleName() + ", hello");
    }
}
  • SimpleMovieLister打上了@Component注解,因此,在Spring的眼中,这是一个bean。
    • movieFinder是这个bean的property
    • 这个property提供了setter方法
  • 所以,这称为在"bean property setter methods"上用@Resource注解注入依赖。

2 打破岁月静好(@Resource takes a name attribute

  • 修改LearnResourceConfig
java 复制代码
@Configuration
public class LearnResourceConfig {
	/*
	MovieFinder这个Bean的名称从movieFinder变成了myMovieFinder
	*/
    @Bean
    public MovieFinder myMovieFinder() {
        return new MovieFinder();
    }
}
  • 无论是字段注入还是setter方法注入,都没法运行了。报错:No bean named 'movieFinder' available

2.1 结论

  • 如果@Resource注解中指定了name属性,那么Spring只会根据name属性的值去找bean,找不到则报错。

2.2 那我不指定呢?【结论:又能正常执行了】

  • 如果@Resource注解没有指定name属性,那么会根据字段名或setter方法推断一个默认名字。
    • 如果根据默认名字找到了bean,那就注入这个bean。
    • 如果根据默认名字找不到bean,那就降级为根据类型去找bean。
    • 如果还找不到,那就报错。

2.2.1 default name

  • org.springframework.context.annotation.CommonAnnotationBeanPostProcessor.ResourceElement#ResourceElement
java 复制代码
public ResourceElement(Member member, AnnotatedElement ae, @Nullable PropertyDescriptor pd) {
		super(member, pd);
		Resource resource = ae.getAnnotation(Resource.class);
		String resourceName = resource.name();
		Class<?> resourceType = resource.type();
		this.isDefaultName = !StringUtils.hasLength(resourceName);
		if (this.isDefaultName) {
			resourceName = this.member.getName();
			if (this.member instanceof Method && resourceName.startsWith("set") && resourceName.length() > 3) {
				resourceName = Introspector.decapitalize(resourceName.substring(3));
			}
		}
		...
  • 如果是字段注入,那么defaultName为字段名。
java 复制代码
@Component
public class SimpleMovieLister {
    @Resource
    private MovieFinder forrestMovieFinder;

	...
}
  • 如果是setter方法注入(setXXX),那么defaultName为XXX
java 复制代码
@Component
public class SimpleMovieLister {
    private MovieFinder movieFinder;

    @Resource
    public void setJerryMovieFinder(MovieFinder movieFinder) {
        this.movieFinder = movieFinder;
    }
    ...
}

3 总结【@Resource是by-name来注入依赖的】

  • 如果@Resource注解中指定了name属性,那么Spring只会根据name属性的值去找bean,找不到则报错。
  • 如果@Resource注解没有指定name属性,那么会根据字段名或setter方法推断一个默认名字。
    • 如果根据默认名字找到了bean,那就注入这个bean。
    • 如果根据默认名字找不到bean,那就降级为根据类型去找bean。
    • 如果还找不到,那就报错。
相关推荐
克拉克盖博1 小时前
chapter03_Bean的实例化与策略模式
java·spring·策略模式
小兔兔吃萝卜6 小时前
Spring 创建 Bean 的 8 种主要方式
java·后端·spring
AAA修煤气灶刘哥8 小时前
面试官: SpringBoot自动配置的原理是什么?从启动到生效,一文讲透
后端·spring·面试
qq_三哥啊10 小时前
【IDEA】设置Debug调试时调试器不进入特定类(Spring框架、Mybatis框架)
spring·intellij-idea·mybatis
别惹CC10 小时前
Spring AI 进阶之路01:三步将 AI 整合进 Spring Boot
人工智能·spring boot·spring
寒士obj10 小时前
Spring事物
java·spring
IT毕设实战小研19 小时前
基于Spring Boot 4s店车辆管理系统 租车管理系统 停车位管理系统 智慧车辆管理系统
java·开发语言·spring boot·后端·spring·毕业设计·课程设计
甄超锋20 小时前
Java ArrayList的介绍及用法
java·windows·spring boot·python·spring·spring cloud·tomcat
Java小白程序员1 天前
Spring Framework:Java 开发的基石与 Spring 生态的起点
java·数据库·spring
甄超锋1 天前
Java Maven更换国内源
java·开发语言·spring boot·spring·spring cloud·tomcat·maven