@Resource依赖注入原理

spring可以通过@Resource注解注入依赖,和@Autowired存在区别:

  1. @Autowired默认按类型注入,而@Resource默认按名称和类型结合注入
  2. @Autowired支持通过属性required设置为非必须注入,当依赖不存在时不会报错;@Resource注解不支持
  3. @Autowired能够配合@Qualifier@Primary@Priority共同使用

那么spring实现@Resource注解注入依赖的原理是什么,现在可以看下,逻辑不复杂,先判断容器中是否包含需要注入的依赖bean实例对象,没有的话就跟@Autowired解析注入的依赖的方式一样,可以看我前一篇**@Autowired依赖注入原理**,存在的话则到容器中根据name和type获取依赖bean实例对象。

java 复制代码
	protected Object autowireResource(BeanFactory factory, LookupElement element, @Nullable String requestingBeanName)
			throws NoSuchBeanDefinitionException {
......
		if (factory instanceof AutowireCapableBeanFactory) {
			AutowireCapableBeanFactory beanFactory = (AutowireCapableBeanFactory) factory;
			DependencyDescriptor descriptor = element.getDependencyDescriptor();
			//容器中没有依赖bean实例对象,则走@Autowired自动注入逻辑
			if (this.fallbackToDefaultTypeMatch && element.isDefaultName && !factory.containsBean(name)) {
				autowiredBeanNames = new LinkedHashSet<>();
				resource = beanFactory.resolveDependency(descriptor, requestingBeanName, autowiredBeanNames, null);
				if (resource == null) {
					throw new NoSuchBeanDefinitionException(element.getLookupType(), "No resolvable resource object");
				}
			}
			//到容器中获取根据name和type获取bean实例对象
			else {
				resource = beanFactory.resolveBeanByName(name, descriptor);
				autowiredBeanNames = Collections.singleton(name);
			}
		}
		else {
			resource = factory.getBean(name, element.lookupType);
			autowiredBeanNames = Collections.singleton(name);
		}
......
		return resource;
	}
	//调用BeanFactory的getBean方法根据name和type获取bean实例对象
	public Object resolveBeanByName(String name, DependencyDescriptor descriptor) {
		InjectionPoint previousInjectionPoint = ConstructorResolver.setCurrentInjectionPoint(descriptor);
		try {
			return getBean(name, descriptor.getDependencyType());
		}
		finally {
			ConstructorResolver.setCurrentInjectionPoint(previousInjectionPoint);
		}
	}

获取到依赖bean实例对象后,就会以反射方式set到字段中,进而完成依赖注入。这里可以看出@Resource获取依赖再注入到字段的逻辑相对简单,虽然使用起来没有@Autowired灵活,但是效率会高很多。

java 复制代码
		protected void inject(Object target, @Nullable String requestingBeanName, @Nullable PropertyValues pvs)
				throws Throwable {

			if (this.isField) {
				Field field = (Field) this.member;
				ReflectionUtils.makeAccessible(field);
				//反射注入依赖到字段
				field.set(target, getResourceToInject(target, requestingBeanName));
			}
			else {
				if (checkPropertySkipping(pvs)) {
					return;
				}
				try {
					Method method = (Method) this.member;
					ReflectionUtils.makeAccessible(method);
					method.invoke(target, getResourceToInject(target, requestingBeanName));
				}
				catch (InvocationTargetException ex) {
					throw ex.getTargetException();
				}
			}
		}

总结下,spring处理通过@Resource注解注入依赖时,会到容器中根据name和type获取依赖bean实例对象,再反射set注入,更多请关注微信公众号 葡萄开源

相关推荐
rocky_rocky28 分钟前
ComboBox的异步延迟加载机制
后端
接着奏乐接着舞30 分钟前
spring cloud知识点
后端·spring·spring cloud
ltl1 小时前
位置编码:为什么需要它,为什么用正弦
后端
明月_清风2 小时前
Go 函数设计的工程智慧:多返回值、闭包与那些"反直觉"的选择
后端·go
却尘2 小时前
一个 `&` 引发的血案:改完配置 pipeline 装聋作哑,顺便重学了 Python/Go/Java
后端·go
倚栏听风雨2 小时前
Spring AI 实战:用 JdbcChatMemory + MySQL 给 AI 接上「长期记忆」
后端
我叫黑大帅2 小时前
最简单的生产-消费者,你都会遇到哪些问题?
后端·面试·go
swipe3 小时前
Agentic RAG:用 LangGraph 构建会路由、会纠错、会收敛的闭环 RAG
后端·langchain·llm
折哥的程序人生 · 物流技术专研4 小时前
《Java 100 天进阶之路》第23篇:缓冲区数据结构 ByteBuffer
java·开发语言·数据结构·后端·面试·求职招聘
还是鼠鼠4 小时前
AI掘金头条新闻系统 (Toutiao News)-获取新闻分类
后端·python·mysql·fastapi·web