七叶笔记 » golang编程 » Spring 依赖注入原理

Spring 依赖注入原理

有时所在项目忙于业务快速迭代,只知道如何应用框架组件,底层原理却忽略了,框架许多精妙设计没掌握到。每天重复流水线工作,业务层面有所提升,但技术原地不动,每次热情高涨探索底层实现,由于种种原因,坚持没多久就放弃了。为了解开放弃探索小伙伴们心中疑惑,本人花了一些时间研究了底层实现。此次分享学习心得为Spring 依赖注入原理,在实际项目中最常见的依赖注入场景,跨业务注解服务互相调用,配置文件引用第三方服务时信息配置。

一 什么是依赖注入

通过类setXxx函数,有参构造函数或注解形式为类的成员属性附上值

二 依赖注入有以下方法

下面这几张注入方式经常碰到,在工作一段时间后,沉下心翻看源码后,多年的疑惑才解开。

1 通过xml配置文件为类实例属性附上值常用有

1.1 bean子标签<property name=”name” value=”test”/> 标签配上值,property标签的name属性值在类中需要对应public权限setXxx函数如下

 XML文件配置
<bean id="dataSource" class="test.Db">
<property name="name" value="QQ"/>
</bean>  
 JAVA类构建
package test;
public class Db {
private String name;
public void setName(String name) {
this.name = name;
}
}  

1.2 bean子标签<constructor-arg name=”name” value=”test”/> 标签配上值,constructor-arg标签的name属性值,在类中构造函数需要对应参数如下

 XML文件配置
<bean id="dataSource" class="test.Db" >
        <constructor-arg name="name" value="qq"/>
</bean>  
 JAVA类构建
public class Db {
private String name;
public Db(String name) {
this.name = name;
}
}  

2 类属性加上属性自动注入注解

 package test;
public class Db {
@Value("${name}")
 private String name;
}
  
 package test;
public class Db {
@Autowired
private X x;
}

package test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Controller;
import org.springframework.stereotype.Service;

@Service
public class X {
}  

三 依赖注入原理

上面的几种形式在实际项目经常碰到,如何实现呢?

1 基于bean子标签property注入原理

1.1 加载bean配置文件,解析bean标签属性值,赋值给BeanDefinition实例并存进集合中,执行finishBeanFactoryInitialization(ConfigurableListableBeanFactory beanFactory),实例化bean(使用反射技术实例化)。

再调用populateBean(String beanName, RootBeanDefinition mbd, @Nullable BeanWrapper bw),内部执行使用反射技术(ReflectionUtils.makeAccessible(writeMethod);

writeMethod.invoke(getWrappedInstance(), value); writeMethod为Method实例),最后调用函数setXxx为属性赋值。

2 基于bean子标签constructor-arg注入原理

2.1 加载bean配置文件,解析bean标签属性值,赋值给BeanDefinition实例并存进集合中,执行finishBeanFactoryInitialization(ConfigurableListableBeanFactory beanFactory),当bean的factoryMethodName属性值为空,bean的有参构造函数多个且调用了。执行函数autowireConstructor(String beanName, RootBeanDefinition mbd, @Nullable Constructor<?>[] ctors, @Nullable Object[] explicitArgs)传入参数。

通过BeanDefinition的成员属性constructorArgument获取构造函数参数,

再执行函数resolveConstructorArguments(String beanName, RootBeanDefinition mbd, BeanWrapper bw,ConstructorArgumentValues cargs, ConstructorArgumentValues resolvedValues)传入参数,循环遍历构造函数实例,获取出匹配的构造函数。

通过反射实例化并通过构造函数参数为类属性赋值(即执行ctor.newInstance(argsWithDefaultValues),ctor为Constructor<T>类型)。

3 基于注解注入待续

上面写得有不对的地方,请小伙伴指出来!也祝小伙们周末愉快!

相关文章