Со временем капли воды и камни изнашиваются 😄
лотерея
Эта статья подала заявку на подарки вокруг Наггетс,сведения о деятельности
1. Способ участия: оставить сообщение в комментариях.
2. Правила комментирования: Обсудите содержание этой статьи. Например, содержание статьи, макет.
3. Программное обеспечение для лотереи:лотерейное программное обеспечениеи введите их в порядке комментариев.
4. Розыгрыш призов:Значок «Наггетс»*2, один игрок может получить не более одного значка «Наггетс».
5. Правила лотереи:По состоянию на 10 сентября в области комментариев взаимодействовало более 10 человек (исключая самого автора), и автор может раздать 2 значка Nuggets по лотерее от своего имени (официально ответственность несут Nuggets).
6. Время розыгрыша:12 сентября. Чтобы обеспечить честность и честность, после крайнего срока 12 сентября я помещу список всех студентов, оставивших сообщение в области комментариев, в лотерейную программу и загружу видео двух счастливчиков, выбранных здесь.
предисловие
Привет, ребята, этоSpring源码
Восьмая статья цикла. Эта статья длинная (5800 слов), собирать и дегустировать рекомендуется не спеша.
Предыдущая статья была проанализированаdoCreateBean
Один из ключевых моментов метода:createBeanInstance
.
Целью этого метода является создание экземпляра Bean. упоминалось выше вcreateBeanInstance
Существует четыре способа создания экземпляра bean-компонента:
- Обратный звонок поставщика:
obtainFromSupplier()
- Инициализация фабричного метода:
instantiateUsingFactoryMethod()
- Конструктор автоматически вводит инициализацию:
autowireConstructor()
- Внедрение конструктора по умолчанию:
instantiateBean()
в предыдущем блогеИсходный код Spring (семь) — поставщик, создание экземпляра фабричного метода Bean-createBeanInstanceпроанализировано вОбратные вызовы поставщиков и фабричные методы, в этой статье будут проанализированы два других метода. до сих пор внутриAbstractAutowireCapableBeanFactory
.createBeanInstance
метод.
// 上述代码省略。。。
boolean resolved = false;
boolean autowireNecessary = false;
//调用getBean()时不传入参数
if(args == null) {
synchronized(mbd.constructorArgumentLock)
// 当作用域为原型时、多次调用getBean()时不传入参数,
//从缓存中获取这段逻辑才会被执行
// resolvedConstructorOrFactoryMethod 缓存了已解析的构造函数或工厂方法
if(mbd.resolvedConstructorOrFactoryMethod != null) {
// resolved为true,表示当前bean的构造方法已经确定了,
//也代表该Bean之前被解析过
resolved = true;
// constructorArgumentsResolved:将构造函数参数标记为已解析
//true就是标记为了已解析
// 默认为 false。
// 如果autowireNecessary为true说明是采用有参构造函数注入
autowireNecessary = mbd.constructorArgumentsResolved;
}
}
}
//resolved = true,表示当前bean的构造方法已经确定了
if(resolved) {
// autowireNecessary = true ,表示采用有参构造函数注入
if(autowireNecessary) {
// 采用有参构造函数注入
return autowireConstructor(beanName, mbd, null, null);
}
else {
// 构造方法已经确定了,但是没有确定构造方法参数
//那就表示没有构造方法参数,用无参的构造方法来实例化bean
return instantiateBean(beanName, mbd);
}
}
Логика приведенного выше кода такова: если нет отображаемого входящего параметра, судитьresolvedConstructorOrFactoryMethod
Не пустой ли он, если он не пустой, то есть кеш, и используется напрямую конструктор parsed. затем согласноautowireNecessary
Параметры, определяющие, следует ли использовать автоматическое внедрение конструктора с параметрами или внедрение конструктора без параметров.
autowireConstructor
Давайте взглянемautowireConstructor
метод, то естьавтоматический ввод параметризованного конструктора. ВойтиConstructorResolver
КатегорияautowireConstructor
метод.
Метод имеет четыре параметра:
- beanName: имя текущего компонента.
- mbd: определение текущего bean-компонента.
- selectedCtors: отфильтрованный список конструкторов. Может быть нулевым.
- manifestArgs: разработчик отображает переданные аргументы при вызове метода getBean.
public BeanWrapper autowireConstructor(String beanName,
RootBeanDefinition mbd, @Nullable Constructor <? > [] chosenCtors,
@Nullable Object[] explicitArgs) {
BeanWrapperImpl bw = new BeanWrapperImpl();
this.beanFactory.initBeanWrapper(bw);
//最终需要使用的构造方法变量
Constructor <? > constructorToUse = null;
ArgumentsHolder argsHolderToUse = null;
//最终需要使用的参数变量
Object[] argsToUse = null;
// getBean()方法指定了构造方法参数
if(explicitArgs != null) {
argsToUse = explicitArgs;
} else {
// 从缓存中获取构造方法和构造方法参数
// 当作用域为原型时并多次调用 getBean()时没有传递参数
// 创建 Bean 是会走这段缓存逻辑。
// 为单例只会从一次,之后的 getBean() 都会从单例池获取。
Object[] argsToResolve = null;
synchronized(mbd.constructorArgumentLock) {
// resolvedConstructorOrFactoryMethod:缓存已解析的构造函数或工厂方法
constructorToUse = (Constructor <? > ) mbd.resolvedConstructorOrFactoryMethod;
// 找到了mbd中缓存的构造方法
// constructorArgumentsResolved:将构造函数参数标记为已解析
//true就是标记为了已解析
if(constructorToUse != null && mbd.constructorArgumentsResolved) {
// resolvedConstructorArguments:获得已完全解析的构造函数参数
//(参数类型已经确定,能够直接进行使用)
// 正常情况下 resolvedConstructorArguments 的值就是 null
argsToUse = mbd.resolvedConstructorArguments;
if(argsToUse == null) {
//获得部分准备好的构造函数参数(该参数的类型是不确定的,需要进行解析)
argsToResolve = mbd.preparedConstructorArguments;
}
}
}
//如果存在构造函数参数,那么则对参数值进行类型转化
//如给定方法的构造函数 Person(int) 则通过此方法后就会把配置中的
// "5“ 转换为 5
//<constructor-arg index="0" value="5"/>
//缓存中的值可能是原始值也可能是最终值
if(argsToResolve != null) {
argsToUse = resolvePreparedArguments(beanName, mbd, bw,
constructorToUse, argsToResolve, true);
}
}
// 如果待使用的构造方法为null,或待使用的构造方法参数为null,也就是没有缓存
// 这个if代码很长,但其实就去找构造方法、构造方法参数
// 并赋值给 constructorToUse、argsToUse
if(constructorToUse == null || argsToUse == null) {
// chosenCtors表示所指定的构造方法
// 没有指定则获取beanClass中的所有的构造方法作为候选者
// 从这些构造方法中选择一个构造方法
Constructor <? > [] candidates = chosenCtors;
if(candidates == null) {
Class <? > beanClass = mbd.getBeanClass();
try {
// mbd.isNonPublicAccessAllowed():默认为true
// getDeclaredConstructors():获得本类所有构造方法
// getConstructors:获得本类的所有公有构造方法
candidates = (mbd.isNonPublicAccessAllowed() ? beanClass.getDeclaredConstructors() :
beanClass.getConstructors());
} catch(Throwable ex) {
throw new BeanCreationException(mbd.getResourceDescription(),
beanName,
"Resolution of declared constructors on bean Class [" +
beanClass.getName() + "] from ClassLoader [" +
beanClass.getClassLoader() + "] failed", ex);
}
}
// 如果只有一个构造方法,并且没有显示指定构造方法参数
// 并且在xml中没有使用constructor-arg标签
// 则需要判断是不是无参构造方法,
// 如果是 则使用无参构造方法进行实例化
if(candidates.length == 1 && explicitArgs == null &&
!mbd.hasConstructorArgumentValues()) {
Constructor <? > uniqueCandidate = candidates[0];
// 判断是不是无参构造方法
if(uniqueCandidate.getParameterCount() == 0) {
synchronized(mbd.constructorArgumentLock) {
// 确定了构造方法之后进行缓存
mbd.resolvedConstructorOrFactoryMethod =
uniqueCandidate;
mbd.constructorArgumentsResolved = true;
mbd.resolvedConstructorArguments = EMPTY_ARGS;
}
//进行实例化
bw.setBeanInstance(instantiate(beanName, mbd,
uniqueCandidate, EMPTY_ARGS));
return bw;
}
}
// 如果入参 chosenCtors不为null,也就是找到了构造方法,或者autowireMode是构造方法自动注入
//则可能要自动选择构造方法
boolean autowiring = (chosenCtors != null || mbd.getResolvedAutowireMode() ==
AutowireCapableBeanFactory.AUTOWIRE_CONSTRUCTOR);
// 记录解析后的构造方法参数值
ConstructorArgumentValues resolvedValues = null;
// minNrOfArgs:表示所有构造方法中,参数个数最少的构造方法的参数个数是多少
int minNrOfArgs;
if(explicitArgs != null) {
minNrOfArgs = explicitArgs.length;
} else {
// 从BeanDefinition中获取所设置的构造方法参数值
// 值来源于 constructor-arg标签中的 index属性的值 这个值可以随便写
ConstructorArgumentValues cargs = mbd.getConstructorArgumentValues();
// 记录解析后的构造方法参数值
resolvedValues = new ConstructorArgumentValues();
// 解析参数个数 值来源于 constructor-arg标签中的 index属性的值 这个值可以随便写
minNrOfArgs = resolveConstructorArguments(beanName, mbd, bw,
cargs, resolvedValues);
}
// 按构造方法的参数个数降序排序,先排序public构造函数,参数降序排列
// 然后排序非public 的构造函数,参数降序排列
AutowireUtils.sortConstructors(candidates);
int minTypeDiffWeight = Integer.MAX_VALUE;
Set < Constructor <? >> ambiguousConstructors = null;
LinkedList < UnsatisfiedDependencyException > causes = null;
// 遍历构造方法,找到一个最合适的
// 先看参数列表最长的构造方法,根据每个参数的参数类型和参数名去找bean
for(Constructor <? > candidate: candidates) {
// 当前构造方法的参数个数
int parameterCount = candidate.getParameterCount();
// 已经找到选用的构造函数 且该参数个数大于当前遍历的,则不用继续遍历了
// 上面已经按照参数个数降序排列了
if(constructorToUse != null && argsToUse != null && argsToUse.length > parameterCount) {
break;
}
// 在遍历某个构造方法时,如果当前遍历的参数个数小于所指定的参数个数
// 则忽略该构造方法
// minNrOfArgs已经是最小的了,比他还小
// 肯定是不符合我所需要的,就不必往下执行了
if(parameterCount < minNrOfArgs) {
continue;
}
ArgumentsHolder argsHolder;
// 当前遍历到的构造方法的参数类型
Class <? > [] paramTypes = candidate.getParameterTypes();
// 当 getBean()方法没有显示指定构造方法参数,resolvedValues 不为 null
if(resolvedValues != null) {
try {
// 获取参数名
// 查看是否在构造方法上使用@ConstructorProperties
// 注解来定义构造方法参数的名字
String[] paramNames = ConstructorPropertiesChecker.evaluate(
candidate, parameterCount);
if(paramNames == null) {
// ParameterNameDiscoverer用于解析方法、构造函数上的参数名称
ParameterNameDiscoverer pnd = this.beanFactory.getParameterNameDiscoverer();
if(pnd != null) {
//获取构造函数的参数名称
paramNames = pnd.getParameterNames(candidate);
}
}
// 根据当前构造方法的参数类型和参数名从beanFactory中得到bean作为参数值
// resolvedValues:解析后的构造方法参数值
// paramTypes:当前构造方法中每个参数的属性类型
// paramNames:当前构造方法中每个参数的属性名称
// getUserDeclaredConstructor(candidate):获取父类中被重写的构造方法
argsHolder = createArgumentArray(beanName, mbd,
resolvedValues, bw, paramTypes, paramNames,
getUserDeclaredConstructor(candidate), autowiring,
candidates.length == 1);
} catch(UnsatisfiedDependencyException ex) {
// 如果找不到相匹配的,也不会直接报错
// 只能说明当前遍历的构造方法不能用
if(logger.isTraceEnabled()) {
logger.trace("Ignoring constructor [" + candidate +
"] of bean '" + beanName + "': " + ex);
}
// Swallow and try next constructor.
if(causes == null) {
causes = new LinkedList < > ();
}
causes.add(ex);
continue;
}
} else {
// getBean()方法指定了构造方法参数值
// 当前构造方法参数个数与传入的参数个数不相等,跳出本次循环
if(parameterCount != explicitArgs.length) {
continue;
}
// 如果参数个数匹配,则把所有参数值封装为一个ArgumentsHolder对象
argsHolder = new ArgumentsHolder(explicitArgs);
}
// 执行到这里,表示当前构造方法可用,并且也找到了对应的构造方法参数值
// 但是还需要判断,当前构造方法是不是最合适的,也许还有另外的构造方法更合适
// 根据参数类型和参数值计算权重
// Lenient宽松,默认宽松模式是开启的
// 严格模式:解析构造函数时,必须所有的都需要匹配,否则抛出异常
// 宽松模式:使用具有"最接近的模式"进行匹配
int typeDiffWeight = (mbd.isLenientConstructorResolution() ?
argsHolder.getTypeDifferenceWeight(paramTypes) :
argsHolder.getAssignabilityWeight(paramTypes));
// Choose this constructor if it represents the closest match.
// 如果当前构造方法的权重比较小,则表示当前构造方法更合适
// 将当前构造方法和所找到参数值作为待使用的
//遍历下一个构造方法
if(typeDiffWeight < minTypeDiffWeight) {
constructorToUse = candidate;
argsHolderToUse = argsHolder;
argsToUse = argsHolder.arguments;
minTypeDiffWeight = typeDiffWeight;
ambiguousConstructors = null;
} else if(constructorToUse != null && typeDiffWeight ==
minTypeDiffWeight) {
// 如果权重一样,则记录在ambiguousConstructors中,继续遍历下一个构造方法
if(ambiguousConstructors == null) {
ambiguousConstructors = new LinkedHashSet < > ();
ambiguousConstructors.add(constructorToUse);
}
ambiguousConstructors.add(candidate);
}
// 循环结束
}
// 遍历完所有构造方法后,没有找到合适的构造方法,则报错
if(constructorToUse == null) {
if(causes != null) {
UnsatisfiedDependencyException ex = causes.removeLast();
for(Exception cause: causes) {
this.beanFactory.onSuppressedException(cause);
}
throw ex;
}
throw new BeanCreationException(mbd.getResourceDescription(),
beanName, "Could not resolve matching constructor " +
"(hint: specify index/type/name arguments for simple parameters
to avoid type ambiguities)"
);
}
// 如果存在权重一样的构造方法并且不是宽松模式,也报错
// 因为权重一样,Spring不知道该用哪个
// 如果是宽松模式则不会报错,Spring会用找到的第一个
else if(ambiguousConstructors != null &&
!mbd.isLenientConstructorResolution()) {
throw new BeanCreationException(mbd.getResourceDescription(),
beanName, "Ambiguous constructor matches found in bean '" +
beanName + "' " +
"(hint: specify index/type/name arguments for simple parameters
to avoid type ambiguities): " +
ambiguousConstructors);
}
// 如果不是通过getBean()方法指定的参数,那么就把找到的构造方法参数进行缓存
if(explicitArgs == null && argsHolderToUse != null) {
// 缓存找到的构造方法
argsHolderToUse.storeCache(mbd, constructorToUse);
}
}
// 得到了构造方法和构造方法的参数值之后,就可以进行实例化了
Assert.state(argsToUse != null,
"Unresolved constructor arguments");
bw.setBeanInstance(instantiate(beanName, mbd, constructorToUse,
argsToUse));
return bw;
}
Код длинный, но в целом речь идет об определении одной вещи. Определите конструктор, параметры конструктора, а затем вызовитеinstantiate
метод создания экземпляра bean-компонента.
Главная идея:
-
- Сначала проверьте, указаны ли значения конкретного конструктора и параметров конструктора, или кэшируйте значения конкретного конструктора или параметров конструктора в BeanDefinition, и если да, используйте конструктор непосредственно для создания экземпляра.
-
- Если нет определенного конструктора или значения параметра конструктора, он разбивается на следующие процессы:
- А. Если конструктор не определен, найти все конструкторы в классе.
- б) Если есть только один конструктор без аргументов, используйте конструктор без аргументов для прямого создания экземпляра.
- в) Если имеется несколько конструкторов или текущим методом внедрения компонента является автоматическое внедрение конструктора, конструктор должен выбираться автоматически.
- г. Определите минимально необходимое количество значений параметров конструктора в соответствии с указанными значениями параметров конструктора. Если не указано, от
BeanDefinition
изconstructorArgumentValues
приобретение имущества. - д. Отсортируйте все конструкторы в порядке убывания количества предпочтительных параметров общедоступных конструкторов и в порядке убывания параметров непубличных конструкторов.
- е. Перебрать каждый конструктор.
- г. Если звоните
getBean
заданное значение параметра конструктора не отображается, то тип параметра конструктора, имя параметра конструктора и проанализированное значение параметра конструктора получаются в соответствии с текущим циклическим конструктором (resolvedValues
) соответствовать, строитьArgumentsHolder
объект. - h. Если значение параметра конструктора указано при вызове метода getBean, он будет напрямую использовать входящее значение параметра конструктора для создания
ArgumentsHolder
объект. - i. Если соответствующее значение параметра метода построения найдено в соответствии с текущим методом построения, то этот метод построения доступен, но этот метод построения не обязательно является лучшим, поэтому он будет включать в себя наличие нескольких методов построения, соответствующих одному и тому же значению. , в это время для оценки будет использоваться степень соответствия значения и типа конструктора, и будет найдено наилучшее совпадение (чем ниже оценка, тем выше приоритет).
Почему чем ниже оценка, тем выше приоритет?
В основном, чтобы вычислить, насколько хорошо совпадают типы параметров найденного bean-компонента и конструктора.
Предположим, что тип bean-компонента — A, родительский класс A — B, родительский класс B — C, а A реализует интерфейс D.
- Если тип параметра конструктора — A, то есть точное совпадение и оценка равна 0.
- Если тип параметра конструктора — B, то оценка равна 2.
- Если тип параметра конструктора C, то оценка равна 4
- Оценка 1, если тип параметра конструктора D
Вы можете напрямую использовать следующий код для тестирования:
Object[] objects = new Object[]{new A()};
// 0
System.out.println(MethodInvoker.getTypeDifferenceWeight(new Class[]{A.class}, objects));
// 2
System.out.println(MethodInvoker.getTypeDifferenceWeight(new Class[]{B.class}, objects));
// 4
System.out.println(MethodInvoker.getTypeDifferenceWeight(new Class[]{C.class}, objects));
// 1
System.out.println(MethodInvoker.getTypeDifferenceWeight(new Class[]{D.class}, objects));
Получите вышеуказанные знания от图灵学院
.
Пример
Дорогие друзья, вы можете спуститься и запустить пример и прервать суть.
Пример один,
- 1. Создайте классы A и B, где класс A имеет атрибуты B и атрибуты имени. И в
A(B b)
Этот конструктор добавляет@Autowired
Аннотация, указывающая, что объект должен быть создан с помощью этого конструктора.
public class A {
private B b;
private String name;
public A() {
}
@Autowired
public A(B b) {
this.b = b;
}
public A(String name) {
this.name = name;
}
public A(B b, String name) {
this.b = b;
this.name = name;
}
public B getB() {
return b;
}
public void setB(B b) {
this.b = b;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "A{" +
"b=" + b +
", name='" + name + '\'' +
'}';
}
}
public class B {
}
- 2. весна-config4.xml
<?xml version="1.0" encoding="UTF-8" ?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
https://www.springframework.org/schema/context/spring-context.xsd">
<bean class="com.gongj.create.A" id="aa"></bean>
<bean class="com.gongj.create.B" id="bb"></bean>
<context:component-scan base-package="com.gongj.create">
</context:component-scan>
</beans>
- 3. Старт
public static void main(String[] args) {
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext(
"spring-config4.xml");
A a = (A) context.getBean("aa");
System.out.println(a);
}
结果:A{b=com.gongj.create.B@27808f31, name='null'}
Пример второй,
- 1. Измените содержимое конфигурации spring-config4.xml.
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
https://www.springframework.org/schema/context/spring-context.xsd">
<!--这里使用 constructor-arg-->
<bean class="com.gongj.create.A" id="aa">
<constructor-arg index="1" value="gongj"></constructor-arg>
<constructor-arg index="0" ref="bb"></constructor-arg>
</bean>
<bean class="com.gongj.create.B" id="bb"></bean>
<context:component-scan base-package="com.gongj.create">
</context:component-scan>
</beans>
- 2. Старт
public static void main(String[] args) {
ClassPathXmlApplicationContext context =
new ClassPathXmlApplicationContext("spring-config4.xml");
A a = (A)context.getBean("aa");
System.out.println(a);
}
Можете ли вы догадаться, каков результат? Результат - ошибка!
Этот маленький Джей@Autowired
Аннотации отмечены вA(B b, String name)
С точки зрения метода строительства, каков результат?
@Autowired
public A(B b, String name) {
this.b = b;
this.name = name;
}
结果:A{b=com.gongj.create.B@3e57cd70, name='gongj'}
Пример третий,
- 1. Измените область действия A на
prototype
.
<bean class="com.gongj.create.A" id="aa" scope="prototype">
<constructor-arg index="0" ref="bb"></constructor-arg>
<constructor-arg index="1" value="gongj"></constructor-arg>
</bean>
- 2. Старт
public static void main(String[] args) {
ClassPathXmlApplicationContext context =
new ClassPathXmlApplicationContext("spring-config4.xml");
A a = (A)context.getBean("aa");
System.out.println(a+ "-===" + a.hashCode());
A a2 = (A)context.getBean("aa",new B(),"uanjfjef");
System.out.println(a2+ "-===" + a2.hashCode());
A a3 = (A)context.getBean("aa");
System.out.println(a3 + "-===" + a3.hashCode());
}
结果:
A{b=com.gongj.create.B@3e57cd70, name='gongj'}-===161960012
//用了传入的值
A{b=com.gongj.create.B@2c039ac6, name='uanjfjef'}-===1484594489
// 用了缓存的值,而且并没有缓存传入的值
A{b=com.gongj.create.B@3e57cd70, name='gongj'}-===1489069835
Вот несколько способов, чтобы кратко упомянуть:
resolveConstructorArguments
Разберите значения параметров конструктора. Бобы могут быть созданы. Возьмите пример 2 выше.
private int resolveConstructorArguments(String beanName, RootBeanDefinition mbd,
BeanWrapper bw, ConstructorArgumentValues cargs, ConstructorArgumentValues resolvedValues) {
// 获得当前 beanFatory 类型转换器
TypeConverter customConverter = this.beanFactory.getCustomTypeConverter();
// 获得当前 beanFatory 类型转换器为 null,则使用 bw,bw 实现了 TypeConverter
TypeConverter converter = (customConverter != null ? customConverter : bw);
//为给定的BeanFactory和BeanDefinition创建一个BeanDefinitionValueResolver
BeanDefinitionValueResolver valueResolver = new BeanDefinitionValueResolver(
this.beanFactory, beanName, mbd, converter);
// 获得 constructor-arg 标签的个数
int minNrOfArgs = cargs.getArgumentCount();
// 先遍历 cargs 中的indexedArgumentValues
// indexedArgumentValues存的是某个index对应的构造方法参数值
for(Map.Entry < Integer, ConstructorArgumentValues.ValueHolder > entry:
cargs.getIndexedArgumentValues().entrySet()) {
int index = entry.getKey();
if(index < 0) {
throw new BeanCreationException(mbd.getResourceDescription(), beanName,
"Invalid constructor argument index: " + index);
}
if(index > minNrOfArgs) {
minNrOfArgs = index + 1;
}
// 获得构造方法参数值
ConstructorArgumentValues.ValueHolder valueHolder = entry.getValue();
if(valueHolder.isConverted()) {
// 把该数据添加到 resolvedValues 对象中
resolvedValues.addIndexedArgumentValue(index, valueHolder);
} else {
// 把“值”转化为对应的类型
// 获得 constructor-arg 的 value
// resolveValueIfNecessary:这里可能会创建bean 因为 constructor-arg 里面有个 ref 属性
//可以引用其他bean。
Object resolvedValue = valueResolver.resolveValueIfNecessary(
"constructor argument", valueHolder.getValue());
ConstructorArgumentValues.ValueHolder resolvedValueHolder = new ConstructorArgumentValues
.ValueHolder(resolvedValue, valueHolder.getType(), valueHolder.getName());
resolvedValueHolder.setSource(valueHolder);
// 把该数据添加到 resolvedValues 对象中
resolvedValues.addIndexedArgumentValue(index, resolvedValueHolder);
}
}
// for循环结束=====
// 把genericArgumentValues中的值进行类型转化然后添加到resolvedValues中去
// 根据上述循环逻辑差不多
for(ConstructorArgumentValues.ValueHolder valueHolder: cargs.getGenericArgumentValues()) {
if(valueHolder.isConverted()) {
resolvedValues.addGenericArgumentValue(valueHolder);
} else {
Object resolvedValue = valueResolver.resolveValueIfNecessary(
"constructor argument", valueHolder.getValue());
ConstructorArgumentValues.ValueHolder resolvedValueHolder = new ConstructorArgumentValues
.ValueHolder(resolvedValue, valueHolder.getType(), valueHolder.getName());
resolvedValueHolder.setSource(valueHolder);
resolvedValues.addGenericArgumentValue(resolvedValueHolder);
}
}
// for循环结束-------
return minNrOfArgs;
}
Сначала получите текущий Bean'sconstructor-arg
количество тегов minNrOfArgs
, затем перебираетindexedArgumentValues
элементы, получитьkey
(то есть,constructor-arg
помеченindex
элемент), сminNrOfArgs
Сравнивать. Кстатиkey
а такжеvalue
добавить вresolvedValues
в объекте.
createArgumentArray
Держатель параметра конструкции. Имеет окончательные параметры конструкции, которые будут использоваться.
private ArgumentsHolder createArgumentArray(String beanName, RootBeanDefinition mbd,
@Nullable ConstructorArgumentValues resolvedValues, BeanWrapper bw,
Class <?> [] paramTypes, @Nullable String[] paramNames, Executable executable,
boolean autowiring, boolean fallback) throws UnsatisfiedDependencyException {
TypeConverter customConverter = this.beanFactory.getCustomTypeConverter();
TypeConverter converter = (customConverter != null ? customConverter : bw);
// 根据参数类型的个数初始化出来对应的ArgumentsHolder
ArgumentsHolder args = new ArgumentsHolder(paramTypes.length);
Set <ConstructorArgumentValues.ValueHolder> usedValueHolders =
new HashSet<>(paramTypes.length);
Set <String> autowiredBeanNames = new LinkedHashSet <> (4);
// 循环执行
for(int paramIndex = 0; paramIndex < paramTypes.length; paramIndex++) {
// 获得参数类型
Class <? > paramType = paramTypes[paramIndex];
// 获得参数名称
String paramName = (paramNames != null ? paramNames[paramIndex] : "");
// Try to find matching constructor argument value, either indexed or generic.
// 尝试查找匹配的构造函数参数值,无论是索引的还是通用的
ConstructorArgumentValues.ValueHolder valueHolder = null;
// 如果指定了构造方法参数值,那么则看当前paramType有没有对应的值
if(resolvedValues != null) {
// 拿到第paramIndex位置的构造方法参数值,会根据传进去的类型、名称进行匹配
//如果类型、名称不相等则返回null
valueHolder = resolvedValues.getArgumentValue(paramIndex, paramType,
paramName, usedValueHolders);
// 如果找不到直接匹配项,尝试一个通用的,无类型的参数值作为后备,
// 类型转换后可以匹配(例如,String-> int)。
if(valueHolder == null && (!autowiring || paramTypes.length ==
resolvedValues.getArgumentCount())) {
valueHolder = resolvedValues.getGenericArgumentValue(null, null,
usedValueHolders);
}
}
// 如果找到了对应的值,则进行类型转化,把转化前的值存在args.rawArguments中
// 转化后的值存在args.arguments中
if(valueHolder != null) {
usedValueHolders.add(valueHolder);
Object originalValue = valueHolder.getValue();
Object convertedValue;
if(valueHolder.isConverted()) {
convertedValue = valueHolder.getConvertedValue();
args.preparedArguments[paramIndex] = convertedValue;
} else {
// 为给定的方法或构造函数创建一个新的MethodParameter。
MethodParameter methodParam = MethodParameter.forExecutable(
executable, paramIndex);
try {
//进行类型转换
convertedValue = converter.convertIfNecessary(originalValue,
paramType, methodParam);
} catch(TypeMismatchException ex) {
throw new UnsatisfiedDependencyException(mbd.getResourceDescription(),
beanName, new InjectionPoint(methodParam),
"Could not convert argument value of type [" + ObjectUtils.nullSafeClassName(
valueHolder.getValue()) + "] to required type [" + paramType.getName() +
"]: " + ex.getMessage());
}
Object sourceHolder = valueHolder.getSource();
if(sourceHolder instanceof ConstructorArgumentValues.ValueHolder) {
Object sourceValue = ((ConstructorArgumentValues.ValueHolder) sourceHolder)
.getValue();
// 将 resolveNecessary 变量赋值为 true,该变量会影响
// resolvedConstructorArguments:缓存完全解析的构造函数参数
// preparedConstructorArguments:缓存部分准备好的构造函数参数
// 这两个参数是谁为 null 值
args.resolveNecessary = true;
args.preparedArguments[paramIndex] = sourceValue;
}
}
args.arguments[paramIndex] = convertedValue;
args.rawArguments[paramIndex] = originalValue;
} else {
// 如果上述逻辑,没有找到对应的值。举例中的举例一 就会走该代码分支
// 为给定的方法或构造函数创建一个新的MethodParameter。
MethodParameter methodParam = MethodParameter.forExecutable(executable,
paramIndex);
if(!autowiring) {
throw new UnsatisfiedDependencyException(mbd.getResourceDescription(),
beanName, new InjectionPoint(methodParam),
"Ambiguous argument values for parameter of type [" + paramType.getName() +
"] - did you specify the correct bean references as arguments?");
}
// 如果 autowiring 的值为 true,也就是
// 如果入参 chosenCtors不为null,也就是找到了构造方法
//或者autowireMode是构造方法自动注入,则可能要自动选择构造方法
//boolean autowiring = (chosenCtors != null ||
//mbd.getResolvedAutowireMode() == AutowireCapableBeanFactory.AUTOWIRE_CONSTRUCTOR);
try {
// 该方法内部会调用 resolveDependency 方法,得到一个bean,resolveDependency 方法很重要。
Object autowiredArgument = resolveAutowiredArgument(methodParam,
beanName, autowiredBeanNames, converter, fallback);
args.rawArguments[paramIndex] = autowiredArgument;
args.arguments[paramIndex] = autowiredArgument;
args.preparedArguments[paramIndex] = autowiredArgumentMarker;
// 将 resolveNecessary 变量赋值为 true,该变量会影响
// resolvedConstructorArguments:缓存完全解析的构造函数参数
// preparedConstructorArguments:缓存部分准备好的构造函数参数
// 这两个参数是否为 null
args.resolveNecessary = true;
} catch(BeansException ex) {
throw new UnsatisfiedDependencyException(mbd.getResourceDescription(),
beanName, new InjectionPoint(methodParam), ex);
}
}
}
for(String autowiredBeanName: autowiredBeanNames) {
this.beanFactory.registerDependentBean(autowiredBeanName, beanName);
if(logger.isDebugEnabled()) {
logger.debug("Autowiring by type from bean name '" + beanName +
"' via " + (executable instanceof Constructor ? "constructor" :
"factory method") + " to bean named '" + autowiredBeanName + "'");
}
}
// 最后,返回找到的值
return args;
}
По типу параметра и имени параметра текущего метода построения сопоставляется значение параметра метода построения, полученное в результате вышеуказанного анализа.Если совпадение успешно, значение параметра метода построения возвращается для преобразования типа. Затем назначьте необходимые свойства и, наконец, вернитесь.
instantiate
instantiate
метод, который находится вSimpleInstantiationStrategy
реализованы в классе.
@Override
public Object instantiate(RootBeanDefinition bd, @Nullable String beanName,
BeanFactory owner,
final Constructor <? > ctor, Object...args) {
//bd.hasMethodOverrides():是否为此bean定义了方法重写,方法返回true说明重写了
if(!bd.hasMethodOverrides()) {
// 没有重写,直接使用反射实例化即可
if(System.getSecurityManager() != null) {
// use own privileged to change accessibility (when security is on)
AccessController.doPrivileged((PrivilegedAction < Object > )() - > {
ReflectionUtils.makeAccessible(ctor);
return null;
});
}
// 通过BeanUtils直接使用构造器对象实例化bean
return BeanUtils.instantiateClass(ctor, args);
}
else {
// 重写了
// 生成CGLIB创建的子类对象
return instantiateWithMethodInjection(bd, beanName, owner, ctor, args);
}
}
Если бин не настроенlookup-method
,replaced-method
этикетка или@Lookup
Аннотация, экземпляр Bean создается непосредственно посредством отражения. Если настроено, вам нужно использоватьCGLIB
Делайте динамическое проксирование.
instantiateBean
Используйте конструктор без аргументов для создания экземпляров bean-компонентов.
protected BeanWrapper instantiateBean(final String beanName,
final RootBeanDefinition mbd) {
try {
Object beanInstance;
final BeanFactory parent = this;
if(System.getSecurityManager() != null) {
beanInstance = AccessController.doPrivileged((PrivilegedAction<Object>)
() - >
getInstantiationStrategy().instantiate(mbd, beanName, parent),
getAccessControlContext());
}
else {
beanInstance = getInstantiationStrategy().instantiate(mbd, beanName,
parent);
}
BeanWrapper bw = new BeanWrapperImpl(beanInstance);
initBeanWrapper(bw);
return bw;
}
catch(Throwable ex) {
throw new BeanCreationException(
mbd.getResourceDescription(), beanName,
"Instantiation of bean failed", ex);
}
}
instantiateBean
метод по сравнению сinstantiateUsingFactoryMethod
,autowireConstructor
Метод слишком простой, потому что не имеет параметров, поэтому нет необходимости определять конструктор и параметры конструктора, а можно напрямую вызвать стратегию инстанцирования для инстанцирования.
Следующее введениеdoCreateBean()
Последний метод в методе:determineConstructorsFromBeanPostProcessors
, чтобы определить конструктор.
Обсуждение в конце статьи
Как вы, ребята, изучаете исходный код и как лучше понять исходный код? Добро пожаловать, чтобы оставить сообщение в области комментариев.
- Если у вас есть какие-либо вопросы по этой статье или есть ошибки в этой статье, пожалуйста, оставьте комментарий. Если вы считаете, что эта статья была вам полезна, ставьте лайк и подписывайтесь на нее.
Дорога длинна и покрыта препятствиями, и путешествие придет.Если путешествие не остановится, можно ожидать будущего.
Я программист Xiaojie, официальный аккаунт имеет такое же имя. Добро пожаловать в мой публичный аккаунт WeChat.