Spring源码之手写DI

news2025/1/12 20:44:50

Spring源码之手写DI

我们来回顾一下前面手写IOC的内容。

image.png

一、DI介绍

DI(Dependency injection)依赖注入。对象之间的依赖由容器在运行期决定,即容器动态的将某个依赖注入到对象之中。说直白点就是给Bean对象的成员变量赋值。

在这里我们就需要明白几个问题。

1. 哪些地方会有依赖

  1. 构造参数依赖
  2. 属性依赖

2. 依赖注入的本质是什么?

依赖注入的本质是 赋值。赋值有两种情况

  1. 给有参构造方法赋值
  2. 给属性赋值

3. 参数值、属性值有哪些?

具体赋值有两种情况:直接值和Bean依赖。比如

public clsss Girl{
     public Girl(String name, int age, Clouth clouth){
        private String name;
    	private Integer age;
    	private Clothes clothes;
     }
}

4. 直接赋值有哪些?

  1. 基本数据类型:String、int 等
  2. 数组,集合
  3. map

二、构造注入

我们先来看看构造参数注入的情况应该要如何解决。

1.构造注入分析

我们应该如何定义构造参数的依赖?也就是我们需要通过构造方法来创建实例,然后对应的构造方法我们需要传入对应的参数。如果不是通过IoC来处理,我们可以直接通过如下的代码实现。

    public static void main(String[] args) {
        Clouth clouth  = new Dress();
        Girl girl = new Girl("小龙女", 18, clouth);
    }

我们通过直接赋值的方式就可以了。但是在IOC中我们需要通过反射的方式来进行通用处理。在使用反射操作的时候就需要能获取到对应的构造参数的依赖了,这时我们得分析怎么来存储我们的构造参数的依赖了。构造参数的依赖有两个特点:

  1. 顺序
  2. 数量
  3. 类型

上面例子中构造函数的参数

  1. 小丽, String类型
  2. 20, int 类型
  3. clouth, 自定义的Clouth类型,是一个依赖Bean

参数可以有多个,我们可以通过List集合来存储,这样构造参数的顺序就可以通过控制往List集合中add元素的顺来保证。但是需要注意的是,依赖Bean如何表示呢?

基本类型的值直接添加到集合中就可以了,但是依赖Bean的实例对象可能还没有创建,那么这时我们可以抽象出一个类,用来描述依赖Bean的信息。

2. BeanReference

注意:

BeanReference其实就是用来描述依赖Bean信息的

image.png

/**
 * 用于依赖注入中描述bean依赖
 */
public class BeanReference {

	private String beanName;

	private Class<?> type;

	public BeanReference(Class<?> type) {
		this.type = type;
	}

	public String getBeanName() {
		return beanName;
	}

	public void setBeanName(String beanName) {
		this.beanName = beanName;
	}

	public Class<?> getType() {
		return type;
	}

	public void setType(Class<?> type) {
		this.type = type;
	}
}

可以根据name来依赖,也可以按照Type来依赖。另外还有一点需要考虑,就是如何来区分是直接值还是依赖Bean呢?有了上面的设计其实就很容易判断了。

if ( obj instanceOf BeanReference)

但是下面这种直接值是数组或者集合等,同时容器中的元素是依赖Bean的情况:

import java.io.Serializable;
import java.util.List;

public class Girl implements Serializable {
    private String name;
    private Integer age;
    private List<Clothes> clothesList;

    public Girl(String name, Integer age, List<Clothes> clothesList) {
        this.name = name;
        this.age = age;
        this.clothesList = clothesList;
    }
}

这种情况下元素值仍然可以用BeanReference来处理。Bean工厂在处理时需要遍历替换。

3. BeanDefinition实现

接下来我们看看如何具体的来实现DI基于构造函数参数依赖的相关操作。首先是BeanDefinition定义的相关处理。需要在 BeanDefinition中增加构造参数的获取的方法。

image.png

然后我们需要在默认的实现GenericBeanDefinition中增加对应的方法来处理。

image.png

BeanDefinition接口

import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.util.List;

import org.apache.commons.lang3.StringUtils;

/**
 * bean定义接口
 */
public interface BeanDefinition {

	String SCOPE_SINGLETON = "singleton";

	String SCOPE_PROTOTYPE = "prototype";

	/**
	 * 类
	 */
	Class<?> getBeanClass();

	/**
	 * Scope
	 */
	String getScope();

	/**
	 * 是否单例
	 */
	boolean isSingleton();

	/**
	 * 是否原型
	 */
	boolean isPrototype();

	/**
	 * 工厂bean名
	 */
	String getFactoryBeanName();

	/**
	 * 工厂方法名
	 */
	String getFactoryMethodName();

	/**
	 * 初始化方法
	 */
	String getInitMethodName();

	/**
	 * 销毁方法
	 */
	String getDestroyMethodName();

	boolean isPrimary();

	/**
	 * 校验bean定义的合法性
	 */
	default boolean validate() {
		// 没定义class,工厂bean或工厂方法没指定,则认为不合法。
		if (this.getBeanClass() == null) {
			if (StringUtils.isBlank(getFactoryBeanName()) || StringUtils.isBlank(getFactoryMethodName())) {
				return false;
			}
		}

		// 定义了类,又定义工厂bean,不合法
		if (this.getBeanClass() != null && StringUtils.isNotBlank(getFactoryBeanName())) {
			return false;
		}

		return true;
	}

	/**
	 * 获得构造参数定义
	 */
	List<?> getConstructorArgumentValues();

	/**
	 * 属性依赖
	 */
	List<PropertyValue> getPropertyValues();

	/**
	 * 下面的四个方法是供beanFactory中使用的
	 */
	public Constructor<?> getConstructor();

	public void setConstructor(Constructor<?> constructor);

	public Method getFactoryMethod();

	public void setFactoryMethod(Method factoryMethod);

}

BeanDefinition的通用实现GernericBeanDefinition


import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.util.List;

import org.apache.commons.lang3.StringUtils;

public class GenericBeanDefinition implements BeanDefinition {

	private Class<?> beanClass;

	private String scope = BeanDefinition.SCOPE_SINGLETON;

	private String factoryBeanName;

	private String factoryMethodName;

	private String initMethodName;

	private String destroyMethodName;

	private boolean primary;

	private Constructor<?> constructor;

	private Method factoryMethod;

	private List<?> constructorArgumentValues;

	private List<PropertyValue> propertyValues;

	public void setBeanClass(Class<?> beanClass) {
		this.beanClass = beanClass;
	}

	public void setScope(String scope) {
		if (StringUtils.isNotBlank(scope)) {
			this.scope = scope;
		}
	}

	public void setFactoryBeanName(String factoryBeanName) {
		this.factoryBeanName = factoryBeanName;
	}

	public void setFactoryMethodName(String factoryMethodName) {
		this.factoryMethodName = factoryMethodName;
	}

	public void setInitMethodName(String initMethodName) {
		this.initMethodName = initMethodName;
	}

	public void setDestroyMethodName(String destroyMethodName) {
		this.destroyMethodName = destroyMethodName;
	}

	@Override
	public Class<?> getBeanClass() {
		return this.beanClass;
	}

	@Override
	public String getScope() {
		return this.scope;
	}

	@Override
	public boolean isSingleton() {
		return BeanDefinition.SCOPE_SINGLETON.equals(this.scope);
	}

	@Override
	public boolean isPrototype() {
		return BeanDefinition.SCOPE_PROTOTYPE.equals(this.scope);
	}

	@Override
	public String getFactoryBeanName() {
		return this.factoryBeanName;
	}

	@Override
	public String getFactoryMethodName() {
		return this.factoryMethodName;
	}

	@Override
	public String getInitMethodName() {
		return this.initMethodName;
	}

	@Override
	public String getDestroyMethodName() {
		return this.destroyMethodName;
	}

	public void setPrimary(boolean primary) {
		this.primary = primary;
	}

	@Override
	public boolean isPrimary() {
		return this.primary;
	}

	public List<?> getConstructorArgumentValues() {
		return constructorArgumentValues;
	}

	public void setConstructorArgumentValues(List<?> constructorArgumentValues) {
		this.constructorArgumentValues = constructorArgumentValues;
	}

	public List<PropertyValue> getPropertyValues() {
		return propertyValues;
	}

	public void setPropertyValues(List<PropertyValue> propertyValues) {
		this.propertyValues = propertyValues;
	}

	public Constructor<?> getConstructor() {
		return constructor;
	}

	public void setConstructor(Constructor<?> constructor) {
		this.constructor = constructor;
	}

	public Method getFactoryMethod() {
		return factoryMethod;
	}

	public void setFactoryMethod(Method factoryMethod) {
		this.factoryMethod = factoryMethod;
	}

	@Override
	public String toString() {
		return "GenericBeanDefinition [beanClass=" + beanClass + ", scope=" + scope + ", factoryBeanName="
				+ factoryBeanName + ", factoryMethodName=" + factoryMethodName + ", initMethodName=" + initMethodName
				+ ", destroyMethodName=" + destroyMethodName + "]";
	}

	@Override
	public int hashCode() {
		final int prime = 31;
		int result = 1;
		result = prime * result + ((beanClass == null) ? 0 : beanClass.hashCode());
		result = prime * result + ((destroyMethodName == null) ? 0 : destroyMethodName.hashCode());
		result = prime * result + ((factoryBeanName == null) ? 0 : factoryBeanName.hashCode());
		result = prime * result + ((factoryMethodName == null) ? 0 : factoryMethodName.hashCode());
		result = prime * result + ((initMethodName == null) ? 0 : initMethodName.hashCode());
		result = prime * result + ((scope == null) ? 0 : scope.hashCode());
		return result;
	}

	@Override
	public boolean equals(Object obj) {
		if (this == obj)
			return true;
		if (obj == null)
			return false;
		if (getClass() != obj.getClass())
			return false;
		GenericBeanDefinition other = (GenericBeanDefinition) obj;
		if (beanClass == null) {
			if (other.beanClass != null)
				return false;
		} else if (!beanClass.equals(other.beanClass))
			return false;
		if (destroyMethodName == null) {
			if (other.destroyMethodName != null)
				return false;
		} else if (!destroyMethodName.equals(other.destroyMethodName))
			return false;
		if (factoryBeanName == null) {
			if (other.factoryBeanName != null)
				return false;
		} else if (!factoryBeanName.equals(other.factoryBeanName))
			return false;
		if (factoryMethodName == null) {
			if (other.factoryMethodName != null)
				return false;
		} else if (!factoryMethodName.equals(other.factoryMethodName))
			return false;
		if (initMethodName == null) {
			if (other.initMethodName != null)
				return false;
		} else if (!initMethodName.equals(other.initMethodName))
			return false;
		if (scope == null) {
			if (other.scope != null)
				return false;
		} else if (!scope.equals(other.scope))
			return false;
		return true;
	}

}

定义后我们来测试一下对应的应用,定义个TestOneBean,依赖了TestTwoBean

public class TestOneBean {

	private String name;

	private TestTwoBean testTwoBean;

	public TestOneBean(String name, TestTwoBean testTwoBean) {
		super();
		this.name = name;
		this.testTwoBean = testTwoBean;
		System.out.println("调用了含有TestTwoBean参数的构造方法");
	}

	public TestOneBean(String name, TestThreeBean testThreeBean) {
		super();
		this.name = name;
		this.testThreeBean = testThreeBean;
		System.out.println("调用了含有TestThreeBean参数的构造方法");
	}

	public ABean(TestTwoBean testTwoBean) {
		super();
		this.testTwoBean = testTwoBean;
	}

	public void doSomthing() {
		System.out.println(String.format("TestOneBean.doSomthing(): %s TestOneBean.name= %s", this.name, this.testTwoBean.getName()));
	}

	public void init() {
		System.out.println("TestOneBean.init() 执行了");
	}

	public void destroy() {
		System.out.println("TestOneBean.destroy() 执行了");
	}
}

然后在实例化时我们需要做相关的绑定

GenericBeanDefinition bd = new GenericBeanDefinition();
bd.setBeanClass(TestOneBean.class);
// 定义的构造参数的依赖
List<Object> args = new ArrayList<>();
args.add("testOneBean");
// 依赖Bean 通过BeanReference来处理
args.add(new BeanReference("testTwoBean"));
bd.setConstructorArgumentValues(args);
bf.registerBeanDefinition("testOneBean", bd);

构造参数传递后,接下来需要在 BeanFactory中来实现构造参数的注入了

4.BeanFactory实现

前面我们在BeanFactory中实现Bean对象的创建有几种方式

  1. 构造方法创建
  2. 工厂静态方法
  3. 工厂成员方法
import java.util.Map;

public interface BeanFactory {

	Object getBean(String name) throws Exception;

	<T> T getBean(Class<T> type)throws Exception;

	<T> Map<String,T> getBeansOfType(Class<T> type)throws Exception;

	Class<?> getType(String name) throws Exception;
}

BeanFactory的默认实现DefaultBeanFactory

import lombok.extern.slf4j.Slf4j;
import org.apache.commons.collections4.CollectionUtils;
import org.apache.commons.lang3.StringUtils;

import java.io.Closeable;
import java.io.IOException;
import java.lang.reflect.*;
import java.util.*;
import java.util.Map.Entry;
import java.util.concurrent.ConcurrentHashMap;

/**
 * 1.构造参数依赖注入的: 参数真实值获取实现,构造方法方式的构造方法的判定实现
 * 2.缓存原型bean的构造方法、工厂方法,增加了 静态工厂方法、工厂bean工厂方法的参数依赖实现
 * 3.构造循环依赖检测
 * 4.属性依赖实现
 */
@Slf4j
public class DefaultBeanFactory implements BeanFactory, BeanDefinitionRegistry, Closeable {

    protected Map<String, BeanDefinition> beanDefintionMap = new ConcurrentHashMap<>(256);

    private final Map<String, Object> singletonBeanMap = new ConcurrentHashMap<>(256);

    private final Map<Class<?>, Set<String>> typeMap = new ConcurrentHashMap<>(256);

    private final ThreadLocal<Set<String>> buildingBeansRecorder = new ThreadLocal<>();

    /**
     * Spring 中属性依赖的情况: 如果循环依赖中不存在单例,则不可以,否则可以
     */
    private final ThreadLocal<Map<String, Object>> earlyExposeBuildingBeans = new ThreadLocal<>();

    @Override
    public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)
            throws BeanDefinitionRegistException {
        Objects.requireNonNull(beanName, "注册bean需要指定beanName");
        Objects.requireNonNull(beanDefinition, "注册bean需要指定beanDefinition");

        // 校验给入的bean是否合法
        if (!beanDefinition.validate()) {
            throw new BeanDefinitionRegistException("名字为[" + beanName + "] 的bean定义不合法:" + beanDefinition);
        }

        if (this.containsBeanDefinition(beanName)) {
            throw new BeanDefinitionRegistException(
                    "名字为[" + beanName + "] 的bean定义已存在:" + this.getBeanDefinition(beanName));
        }

        this.beanDefintionMap.put(beanName, beanDefinition);
    }

    public void registerTypeMap() throws Exception {
        for (String name : this.beanDefintionMap.keySet()) {
            Class<?> type = this.getType(name);
            //映射本类
            this.registerTypeMap(name, type);
            //父类
            this.registerSuperClassTypeMap(name, type);
            //接口
            this.registerInterfaceTypeMap(name, type);
        }

    }

    private void registerInterfaceTypeMap(String name, Class<?> type) {
        Class<?>[] interfaces = type.getInterfaces();
        for (Class<?> interfaceClasses : interfaces) {
            this.registerTypeMap(name, interfaceClasses);
            //递归找父接口
            this.registerInterfaceTypeMap(name, interfaceClasses);
        }
    }

    private void registerSuperClassTypeMap(String name, Class<?> type) {
        Class<?> superClass = type.getSuperclass();
        if (superClass != null && !superClass.equals(Object.class)) {
            this.registerTypeMap(name, superClass);
            //递归找父类
            this.registerSuperClassTypeMap(name, superClass);
            //找父类实现的接口注册
            this.registerInterfaceTypeMap(name, superClass);
        }
    }

    private void registerTypeMap(String name, Class<?> type) {
        Set<String> names2type = this.typeMap.computeIfAbsent(type, k -> new HashSet<>());
        names2type.add(name);
    }

    @Override
    public Class<?> getType(String name) throws Exception {
        BeanDefinition beanDefinition = this.getBeanDefinition(name);
        Class<?> type = beanDefinition.getBeanClass();
        if (type != null) {
            if (StringUtils.isBlank(beanDefinition.getFactoryMethodName())) {
                // 构造方法来构造对象的,Type就是beanClass,不需做什么。
            } else {
                // 静态工厂方法方式的,反射获得Method,再获取Method的返回值类型
                type = type.getDeclaredMethod(beanDefinition.getFactoryMethodName(), null).getReturnType();
            }
        } else {
            // 获得工厂Bean的Class
            type = this.getType(beanDefinition.getFactoryBeanName());
            // 获取工厂方法的返回值类型
            type = type.getDeclaredMethod(beanDefinition.getFactoryMethodName(), null).getReturnType();
        }

        return type;
    }

    @Override
    public BeanDefinition getBeanDefinition(String beanName) {
        return this.beanDefintionMap.get(beanName);
    }

    @Override
    public boolean containsBeanDefinition(String beanName) {

        return this.beanDefintionMap.containsKey(beanName);
    }

    @Override
    public Object getBean(String name) throws Exception {
        return this.doGetBean(name);
    }

    @Override
    public <T> T getBean(Class<T> type) throws Exception {
        /*
         * 逻辑:
         * 1. 获取其对应的所有的BeanDefinition
         * 2. 如果只有一个,直接获取bean实例返回,否则
         * 3. 遍历找出Primary的
         * 4. 如果primary没有,或大于1个,抛出异常
         * 5. 返回Primary的实例
         */
        Set<String> names = this.typeMap.get(type);
        if (names != null) {
            if (names.size() == 1) {
                return (T) this.getBean(names.iterator().next());
            } else {
                //找Primary
                BeanDefinition beanDefinition = null;
                String primaryName = null;
                StringBuilder nameStrings = new StringBuilder();
                for (String name : names) {
                    beanDefinition = this.getBeanDefinition(name);
                    if (beanDefinition != null && beanDefinition.isPrimary()) {
                        if (primaryName != null) {
                            String mess = type + " 类型的Bean存储多个Primary[" + primaryName + "," + name + "]";
                            log.error(mess);
                            throw new Exception(mess);
                        } else {
                            primaryName = name;
                        }
                    }
                    nameStrings.append(" ").append(name);
                }

                if (primaryName != null) {
                    return (T) this.getBean(primaryName);
                } else {
                    String mess = type + " 类型的Bean存在多个[" + nameStrings + "] 但无法确定Primary";
                    log.error(mess);
                    throw new Exception(mess);
                }
            }
        }
        return null;
    }

    @Override
    public <T> Map<String, T> getBeansOfType(Class<T> type) throws Exception {
        Set<String> names = this.typeMap.get(type);
        if (names != null) {
            Map<String, T> map = new HashMap<>();
            for (String name : names) {
                map.put(name, (T) this.getBean(name));
            }
            return map;
        }
        return null;
    }

    protected Object doGetBean(String beanName) throws Exception {
        Objects.requireNonNull(beanName, "beanName不能为空");

        Object instance = singletonBeanMap.get(beanName);

        if (instance != null) {
            return instance;
        }

        instance = this.getFromEarlyExposeBuildingBeans(beanName);
        if (instance != null) { //这是属性依赖时的循环引用,返回提前暴露的实例
            return instance;
        }

        BeanDefinition beanDefinition = this.getBeanDefinition(beanName);
        Objects.requireNonNull(beanDefinition, "beanDefinition不能为空");

        // 检测构造参数循环依赖
        Set<String> buildingBeans = this.buildingBeansRecorder.get();
        if (buildingBeans == null) {
            buildingBeans = new HashSet<>();
            this.buildingBeansRecorder.set(buildingBeans);
        }
        if (buildingBeans.contains(beanName)) {
            throw new Exception(beanName + " 循环依赖!" + buildingBeans);
        }
        // 记录正在创建的Bean
        buildingBeans.add(beanName);

        if (beanDefinition.isSingleton()) { //如果是单例
            synchronized (this.singletonBeanMap) { //加锁
                instance = this.singletonBeanMap.get(beanName);
                if (instance == null) {//第二次检查
                    instance = doCreateInstance(beanName, beanDefinition);
                    this.singletonBeanMap.put(beanName, instance);
                }
            }
        } else {
            instance = doCreateInstance(beanName, beanDefinition);
        }

        // 创建好实例后,移除创建中记录
        buildingBeans.remove(beanName);

        return instance;
    }

    private Object getFromEarlyExposeBuildingBeans(String beanName) {
        Map<String, Object> earlyExposeBuildingBeansMap = earlyExposeBuildingBeans.get();
        return earlyExposeBuildingBeansMap == null ? null : earlyExposeBuildingBeansMap.get(beanName);
    }

    private Object doCreateInstance(String beanName, BeanDefinition beanDefinition) throws Exception {
        Class<?> type = beanDefinition.getBeanClass();
        Object instance;
        if (type != null) {
            if (StringUtils.isBlank(beanDefinition.getFactoryMethodName())) {
                // 构造方法来构造对象
                instance = this.createInstanceByConstructor(beanDefinition);
            } else {
                // 静态工厂方法
                instance = this.createInstanceByStaticFactoryMethod(beanDefinition);
            }
        } else {
            // 工厂bean方式来构造对象
            instance = this.createInstanceByFactoryBean(beanDefinition);
        }

        this.doEarlyExposeBuildingBeans(beanName, instance);

        // 给入属性依赖
        this.setPropertyDIValues(beanDefinition, instance);

        this.removeEarlyExposeBuildingBeans(beanName, instance);

        // 执行初始化方法
        this.doInit(beanDefinition, instance);

        return instance;
    }

    private void removeEarlyExposeBuildingBeans(String beanName, Object instance) {
        earlyExposeBuildingBeans.get().remove(beanName);
    }

    private void doEarlyExposeBuildingBeans(String beanName, Object instance) {
        Map<String, Object> earlyExposeBuildingBeansMap = earlyExposeBuildingBeans.get();
        if (earlyExposeBuildingBeansMap == null) {
            earlyExposeBuildingBeansMap = new HashMap<>();
            earlyExposeBuildingBeans.set(earlyExposeBuildingBeansMap);
        }
        earlyExposeBuildingBeansMap.put(beanName, instance);
    }

    // 给入属性依赖
    private void setPropertyDIValues(BeanDefinition bd, Object instance) throws Exception {
        if (CollectionUtils.isEmpty(bd.getPropertyValues())) {
            return;
        }
        for (PropertyValue pv : bd.getPropertyValues()) {
            if (StringUtils.isBlank(pv.getName())) {
                continue;
            }
            Class<?> clazz = instance.getClass();
            Field p = clazz.getDeclaredField(pv.getName());

            p.setAccessible(true);
            p.set(instance, this.getOneArgumentRealValue(pv.getValue()));

        }
    }

    /**
     * 构造方法来构造对象
     *
     * @param beanDefinition bean定义
     * @return bean实例对象
     * @throws Exception 异常
     */
    private Object createInstanceByConstructor(BeanDefinition beanDefinition)
            throws Exception {
        /*
        构造参数依赖注入,这里需要做些什么?
        1 得到真正的参数值,因为
        List<?> constructorArgumentValues = beanDefinition.getConstructorArgumentValues();
        constructorArgumentValues 中可能有 BeanReference
        */
        Object[] args = this.getConstructorArgumentValues(beanDefinition);

        // 2 判定应该调用哪个构造方法来创建实例
        return this.determineConstructor(beanDefinition, args).newInstance(args);
    }


    private Object[] getConstructorArgumentValues(BeanDefinition bd) throws Exception {
        List<?> constructorArgumentValues = bd.getConstructorArgumentValues();
        if (CollectionUtils.isEmpty(constructorArgumentValues)) {
            return null;
        }

        Object[] values = new Object[constructorArgumentValues.size()];
        int i = 0;
        for (Object originalValue : constructorArgumentValues) {
            values[i++] = getOneArgumentRealValue(originalValue); //获取真正参数值的逻辑应该是怎样的?
        }
        return values;
    }

    private Object getOneArgumentRealValue(Object originalValue) throws Exception {
        //获取真正参数值,主要是处理BeanReference,得到真正的Bean实例
        Object realValue = null;
        if (originalValue != null) {
            if (originalValue instanceof BeanReference) {
                BeanReference beanReference = (BeanReference) originalValue;
                if (StringUtils.isNotBlank(beanReference.getBeanName())) {
                    realValue = this.getBean(beanReference.getBeanName());
                } else if (beanReference.getType() != null) {
                    realValue = this.getBean(beanReference.getType());
                }
            } else if (originalValue instanceof Object[]) {
                // 处理数组中的bean引用
                Object[] originalValueNew = (Object[]) originalValue;
                for (int i = 0; i < originalValueNew.length; i++) {
                    Object attr = originalValueNew[i];
                    Class<?> attrClass = attr.getClass();
                    String simpleName = attrClass.getSimpleName();
                    BeanReference beanReference = new BeanReference(attrClass);
                    beanReference.setBeanName(getBeanName(simpleName));
                    originalValueNew[i] = beanReference;
                }
                realValue = originalValueNew;
            } else if (originalValue instanceof Collection) {
                // 处理集合中的bean引用
                Type genericSuperclass = originalValue.getClass().getGenericSuperclass();
                Class<? extends Type> genericClass = genericSuperclass.getClass();
                BeanReference beanReference = new BeanReference(genericClass);
                String simpleName = genericClass.getSimpleName();
                String beanName = getBeanName(simpleName);
                beanReference.setBeanName(beanName);
                Collection<BeanReference> originalValueNew = new ArrayList<>();
                originalValueNew.add(beanReference);
                realValue = originalValueNew;
            } else if (originalValue instanceof Properties) {
                // 处理properties中的bean引用
                Properties originValueNew = new Properties();
                originValueNew.put(new BeanReference(null), new BeanReference(null));
                realValue = originValueNew;
            } else if (originalValue instanceof Map) {
                // 处理Map中的bean引用
                Map<BeanReference, BeanReference> originValueNew = new HashMap<>();
                originValueNew.put(new BeanReference(null), new BeanReference(null));
                realValue = originValueNew;
            } else {
                realValue = originalValue;
            }
        }
        return realValue;
    }

    private String getBeanName(String simpleName) {
        if (simpleName == null) {
            throw new IllegalArgumentException("simpleName不能为空");
        }
        String firstChar = simpleName.charAt(0) + "";
        return firstChar.toLowerCase() + simpleName.substring(1);
    }

    private Constructor<?> determineConstructor(BeanDefinition bd, Object[] args) throws Exception {
        /*判定构造方法的逻辑应是怎样的?
        1 先根据参数的类型进行精确匹配查找,如未找到,则进行第2步查找;
        2获得所有的构造方法,遍历,通过参数数量过滤,再比对形参类型与实参类型。
        * */

        Constructor<?> ct = null;

        //没有参数,则用无参构造方法
        if (args == null) {
            return bd.getBeanClass().getConstructor(null);
        }

        // 对于原型bean,从第二次开始获取bean实例时,可直接获得第一次缓存的构造方法。
        ct = bd.getConstructor();
        if (ct != null) {
            return ct;
        }

        // 1 根据参数类型获取精确匹配的构造方法
        Class<?>[] paramTypes = new Class[args.length];
        int j = 0;
        for (Object p : args) {
            paramTypes[j++] = p.getClass();
        }
        try {
            ct = bd.getBeanClass().getConstructor(paramTypes);
        } catch (Exception e) {
            // 这个异常不需要处理
        }

        if (ct == null) {
            // 2 没有精确参数类型匹配的,获得所有的构造方法,遍历,通过参数数量过滤,再比对形参类型与实参类型。
            // 判断逻辑:先判断参数数量,再依次比对形参类型与实参类型
            outer:
            for (Constructor<?> ct0 : bd.getBeanClass().getConstructors()) {
                Class<?>[] paramterTypes = ct0.getParameterTypes();
                if (paramterTypes.length == args.length) {   //通过参数数量过滤
                    for (int i = 0; i < paramterTypes.length; i++) { //再依次比对形参类型与实参类型是否匹配
                        if (!paramterTypes[i].isAssignableFrom(args[i].getClass())) {
                            continue outer; //参数类型不可赋值(不匹配),跳到外层循环,继续下一个
                        }
                    }

                    ct = ct0;  //匹配上了
                    break outer;
                }
            }
        }

        if (ct != null) {
            // 对于原型bean,可以缓存找到的构造方法,方便下次构造实例对象。在BeanDefinfition中获取设置所用构造方法的方法。
            // 同时在上面增加从beanDefinition中获取的逻辑。
            if (bd.isPrototype()) {
                bd.setConstructor(ct);
            }
            return ct;
        } else {
            throw new Exception("不存在对应的构造方法!" + bd);
        }
    }


    private Method determineFactoryMethod(BeanDefinition bd, Object[] args, Class<?> type) throws Exception {
       /*判定工厂方法的逻辑同构造方法的判定逻辑
        1 先根据实参的类型进行精确匹配查找,如未找到,则进行第2步查找;
        2 获得所有方法,遍历,通过方法名、参数数量过滤,再比对形参类型与实参类型。
        * */
        String methodName = bd.getFactoryMethodName();

        if (args == null) {
            return type.getMethod(methodName, null);
        }

        Method m = null;
        // 对于原型bean,从第二次开始获取bean实例时,可直接获得第一次缓存的构造方法。
        m = bd.getFactoryMethod();
        if (m != null) {
            return m;
        }

        // 1 先根据实参的类型进行精确匹配查找
        Class[] paramTypes = new Class[args.length];
        int j = 0;
        for (Object p : args) {
            paramTypes[j++] = p.getClass();
        }
        try {
            m = type.getMethod(methodName, paramTypes);
        } catch (Exception e) {
            // 这个异常不需要处理
        }

        if (m == null) {
            // 没有精确参数类型匹配的,则遍历匹配所有的方法
            // 2 获得所有方法,遍历,通过方法名、参数数量过滤,再比对形参类型与实参类型。
            outer:
            for (Method m0 : type.getMethods()) {
                if (!m0.getName().equals(methodName)) {
                    continue;
                }
                Class<?>[] paramterTypes = m.getParameterTypes();
                if (paramterTypes.length == args.length) {
                    for (int i = 0; i < paramterTypes.length; i++) {
                        if (!paramterTypes[i].isAssignableFrom(args[i].getClass())) {
                            continue outer;
                        }
                    }

                    m = m0;
                    break outer;
                }
            }
        }

        if (m != null) {
            // 对于原型bean,可以缓存找到的方法,方便下次构造实例对象。在BeanDefinfition中获取设置所用方法的方法。
            // 同时在上面增加从beanDefinition中获取的逻辑。
            if (bd.isPrototype()) {
                bd.setFactoryMethod(m);
            }
            return m;
        } else {
            throw new Exception("不存在对应的构造方法!" + bd);
        }
    }

    // 静态工厂方法
    private Object createInstanceByStaticFactoryMethod(BeanDefinition bd) throws Exception {

        Object[] realArgs = this.getConstructorArgumentValues(bd);
        Class<?> type = bd.getBeanClass();
        Method m = this.determineFactoryMethod(bd, realArgs, type);
        return m.invoke(type, realArgs);
    }

    // 工厂bean方式来构造对象
    private Object createInstanceByFactoryBean(BeanDefinition bd) throws Exception {

        Object[] realArgs = this.getConstructorArgumentValues(bd);
        Method m = this.determineFactoryMethod(bd, realArgs, this.getType(bd.getFactoryBeanName()));

        Object factoryBean = this.doGetBean(bd.getFactoryBeanName());
        return m.invoke(factoryBean, realArgs);
    }

    /**
     * 执行初始化方法
     *
     * @param beanDefinition bean定义
     * @param instance       实例对象
     * @throws Exception 异常
     */
    private void doInit(BeanDefinition beanDefinition, Object instance) throws Exception {
        // 执行初始化方法
        if (StringUtils.isNotBlank(beanDefinition.getInitMethodName())) {
            Method m = instance.getClass().getMethod(beanDefinition.getInitMethodName(), null);
            m.invoke(instance, null);
        }
    }

    @Override
    public void close() throws IOException {
        // 执行单例实例的销毁方法
        for (Entry<String, BeanDefinition> e : this.beanDefintionMap.entrySet()) {
            String beanName = e.getKey();
            BeanDefinition bd = e.getValue();

            if (bd.isSingleton() && StringUtils.isNotBlank(bd.getDestroyMethodName())) {
                Object instance = this.singletonBeanMap.get(beanName);
                try {
                    Method m = instance.getClass().getMethod(bd.getDestroyMethodName(), null);
                    m.invoke(instance, null);
                } catch (NoSuchMethodException | SecurityException | IllegalAccessException | IllegalArgumentException
                         | InvocationTargetException e1) {
                    log.error("执行bean[" + beanName + "] " + bd + " 的 销毁方法异常!", e1);
                }
            }
        }
    }
}

我们通过构造方法创建其实是通过无参构造方法来处理的,这时我们需要改变这块的逻辑,通过有参构造方法来实现。

// 构造方法来构造对象
private Object createInstanceByConstructor(BeanDefinition bd)
    throws InstantiationException, IllegalAccessException {
    try {
        return bd.getBeanClass().newInstance();
    } catch (SecurityException e1) {
        log.error("创建bean的实例异常,beanDefinition:" + bd, e1);
        throw e1;
    }
}

我们就需要对上面的方法做出改变。

// 构造方法来构造对象
private Object createInstanceByConstructor(BeanDefinition bd)
    throws InstantiationException, IllegalAccessException {
    // 1. 得到真正的参数值
    List<?> constructorArgumentValues = bd.getConstructorArgumentValues(); 
    // 2.根据对应的构造参数依赖获取到对应的 Constructor 
    Constructor  constructor = 得到对应的构造方法
        // 3.用实际参数值调用构造方法创建对应的对象
    return constructor.newInstance(Object ... 实参值); 
}

通过上面的分析我们需要获取对应的构造器。这块我们需要通过反射来获取了。下面是具体的实现逻辑

image.png

根据上面的分析,实现逻辑可分为两步

  1. 先根据参数的类型进行精确匹配查找,如果没有找到,继续执行第二步操作
  2. 获得所有的构造方法,遍历构造方法,通过参数数量过滤,再比对形参与实参的类型

因为这里有个情况,实参是Boy,构造方法的形参是Person,第一种精确匹配就没有办法关联了。

image.png

具体的实现代码如下:

private Constructor<?> determineConstructor(BeanDefinition bd, Object[] args) throws Exception {
    /*判定构造方法的逻辑应是怎样的?
        1 先根据参数的类型进行精确匹配查找,如未找到,则进行第2步查找;
        2获得所有的构造方法,遍历,通过参数数量过滤,再比对形参类型与实参类型。
        * */

    Constructor<?> ct = null;

    //没有参数,则用无参构造方法
    if (args == null) {
        return bd.getBeanClass().getConstructor(null);
    }

    // 1 先根据参数的类型进行精确匹配查找
    Class<?>[] paramTypes = new Class[args.length];
    int j = 0;
    for (Object p : args) {
        paramTypes[j++] = p.getClass();
    }
    try {
        ct = bd.getBeanClass().getConstructor(paramTypes);
    } catch (Exception e) {
        // 这个异常不需要处理
    }

    if (ct == null) {
        // 2 没有精确参数类型匹配的,获得所有的构造方法,遍历,通过参数数量过滤,再比对形参类型与实参类型。
        // 判断逻辑:先判断参数数量,再依次比对形参类型与实参类型
        outer:
        for (Constructor<?> ct0 : bd.getBeanClass().getConstructors()) {
            Class<?>[] paramterTypes = ct0.getParameterTypes();
            if (paramterTypes.length == args.length) {   //通过参数数量过滤
                for (int i = 0; i < paramterTypes.length; i++) { //再依次比对形参类型与实参类型是否匹配
                    if (!paramterTypes[i].isAssignableFrom(args[i].getClass())) {
                        continue outer; //参数类型不可赋值(不匹配),跳到外层循环,继续下一个
                    }
                }

                ct = ct0;  //匹配上了
                break outer;
            }
        }
    }

    if (ct != null) {
        return ct;
    } else {
        throw new Exception("不存在对应的构造方法!" + bd);
    }
}

上面我们考虑的是BeanFactory通过构造器来获取对象的逻辑,那如果我们是通过静态工厂方法或者成员工厂方法的方式来处理的,那么构造参数依赖的处理是否和前面的是一样的呢?其实是差不多的,我们需要根据对应的构造参数来推断对应的工厂方法

// 静态工厂方法
private Object createInstanceByStaticFactoryMethod(BeanDefinition bd) throws Exception {
    Object[] realArgs = this.getConstructorArgumentValues(bd);
    Class<?> type = bd.getBeanClass();
    Method m = this.determineFactoryMethod(bd, realArgs, type);
    return m.invoke(type, realArgs);
}

// 工厂bean方式来构造对象
private Object createInstanceByFactoryBean(BeanDefinition bd) throws Exception {
    Object[] realArgs = this.getConstructorArgumentValues(bd);
    Method m = this.determineFactoryMethod(bd, realArgs, this.getType(bd.getFactoryBeanName()));

    Object factoryBean = this.doGetBean(bd.getFactoryBeanName());
    return m.invoke(factoryBean, realArgs);
}

5.缓存功能

对于上面的处理过程相信大家应该清楚了,我们通过推断也得到了对应的构造方法或者对应的工厂方法,那么我们可以不可以在下次需要再次获取的时候省略掉推导的过程呢?显然我们可以在BeanDefinition中增加缓存方法可以实现这个需求。

image.png

6. 循环依赖问题

image.png

  上图是循环依赖的三种情况,虽然方式有点不一样,但是循环依赖的本质是一样的,就你的完整创建要依赖于我,我的完整创建也依赖于你。相互依赖从而没法完整创建造成失败。

  我们通过构造参数依赖是完全可能出现上面的情况的,那么这种情况我们能解决吗?构造依赖的情况我们是解决不了的。

public class Test01 {

    public static void main(String[] args) {
        new TestService1();
    }
}

class TestService1{
    private TestService2 testService2 = new TestService2();
}

class TestService2{
    private  TestService1 testService1 = new TestService1();
}

既然解决不了,那么我们在程序中如果出现了,应该要怎么来解决呢?

其实我们可以在创建一个Bean的时候记录下这个Bean,当这个Bean创建完成后我们再移除这个Bean,然后我们在getBean的时候判断记录中是否有该Bean,如果有就判断为循环依赖,并抛出异常。数据结构我们可以通过Set集合来处理。

image.png

到此构造注入的实现就告一段落了。

三、属性注入

  上面搞定了构造注入的方式。接下来我们再看看属性注入的方式有什么需要注意的地方。

1. 属性依赖分析

  属性依赖就是某个属性依赖某个值。

public class Girl {

    private String name;
    private int age;
    private List<Clothes> ClothesList;

    // ....
}

  那么在获取实例对象后如何根据相关的配置来给对应的属性来赋值呢?这时我们可以定义一个实体类 PropertyValue来记录相关的属性和值。

image.png

2.BeanDefinition实现

  这时我们就需要在BeanDefinition中关联相关属性信息了。

image.png

3.BeanFactory实现

  然后我们在BeanFactory的默认实现DefaultBeanFactory中实现属性值的依赖注入。

// 创建好实例对象
// 给属性依赖赋值
this.setPropertyDIValues(bd,instance);
// 执行初始化相关方法
this.doInit(bd,instance);

  具体的实现代码如下:

// 给入属性依赖
private void setPropertyDIValues(BeanDefinition bd, Object instance) throws Exception {
    if (CollectionUtils.isEmpty(bd.getPropertyValues())) {
        return;
    }
    for (PropertyValue pv : bd.getPropertyValues()) {
        if (StringUtils.isBlank(pv.getName())) {
            continue;
        }
        Class<?> clazz = instance.getClass();
        Field p = clazz.getDeclaredField(pv.getName());
        //暴力访问  private
        p.setAccessible(true);
        p.set(instance, this.getOneArgumentRealValue(pv.getValue()));

    }
}

4.循环依赖问题

  在构造参数依赖中我们发现没有办法解决,在属性依赖中同样会存在循环依赖的问题,这时我们能解决吗?

image.png

  其实这种循环依赖的情况,不在IOC场景下非常好解决。如下

Boy b = new Boy();
Girl g = new Girl();
b.setGirl(g);
g.setBoy(b);

  但是在IOC好像不是太好解决:

image.png

  针对这种情况我们需要通过 提前暴露来解决这个问题,具体看代码!!!

private void doEarlyExposeBuildingBeans(String beanName, Object instance) {
    Map<String,Object> earlyExposeBuildingBeansMap = earlyExposeBuildingBeans.get();
    if(earlyExposeBuildingBeansMap == null) {
        earlyExposeBuildingBeansMap = new HashMap<>();
        earlyExposeBuildingBeans.set(earlyExposeBuildingBeansMap);
    }
    earlyExposeBuildingBeansMap.put(beanName,instance);
}

最后现阶段已经实现的类图结构

image.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1854217.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

go语言day03

目录 一、 go语言的数据类型&#xff1a; 二、声明赋值的简写形式&#xff1a; ":" 1&#xff09;重复使用的编译错误 2&#xff09;在全局变量中使用 : 会报编译错误 三、变量规则&#xff1a; 0&#xff09;变量的命名规则&#xff1a; 1&#xff09;创建的局部…

西瓜视频基于 Hertz 的微服务落地实践

# 1. 西瓜视频微服务架构设计 ## 1.1 西瓜视频介绍 **西瓜视频**是一个开眼界、涨知识的视频 App&#xff08;Informative Video Platform&#xff09;&#xff0c;作为国内领先的**中长视频**平台&#xff0c;它源源不断地为不同人群提供优质内容&#xff0c;让人们看到更丰…

OpenAPI

大家好我是苏麟 , 今天带来一个前端生成接口的工具 . 官网 : GitHub - ferdikoomen/openapi-typescript-codegen: NodeJS library that generates Typescript or Javascript clients based on the OpenAPI specification 安装命令 npm install openapi-typescript-codegen --sa…

flex布局无法设置图片icon和文本垂直居中对齐问题

项目场景&#xff1a; 需要实现下面的效果&#xff0c;即图标和文字垂直对齐。 问题描述 直接使用flex布局并设置垂直居中&#xff0c;发现并没有垂直对齐&#xff0c;图片明显偏上。 .wrapper {display: flex;align-items: center; }.view-icon {height: 28px;width: 28px;m…

02_ESP32+MicroPython 点亮LED灯

书接第1篇《01_ESP32 MicroPython开发环境搭建_eps32开发板-CSDN博客》 想要让一个引脚输出高电平&#xff0c;只需要找到对应的GPIO然后通过on()或者value(1)操作就可以&#xff0c;同理如果想要输出低电平让LED灯灭&#xff0c;只需要调用off()或者value(0)就行。 一、点亮…

InfoMasker :新型反窃听系统,保护语音隐私

随着智能手机、智能音箱等设备的普及&#xff0c;人们越来越担心自己的谈话内容被窃听。由于这些设备通常是黑盒的&#xff0c;攻击者可能利用、篡改或配置这些设备进行窃听。借助自动语音识别 (ASR) 系统&#xff0c;攻击者可以从窃听的录音中提取受害者的个人信息&#xff0c…

Linux C/C++ socket函数

目录 socket函数 函数原型 头文件 功能 返回值 参数 错误码 socket函数 函数原型 int socket(int domain, int type, int protocol); 头文件 #include <sys/types.h> #include <sys/socket.h> 功能 创建一个用于通信的端点&#xff0c;并返回一个文件描述符…

档案数字化建设花费主要在哪里

在档案数字化建设中&#xff0c;主要花费包括以下几个方面&#xff1a; 1. 技术设备和软件&#xff1a;包括购买和维护服务器、计算机、扫描仪、存储设备等硬件设备&#xff0c;以及购买和使用专久智能档案数字化软件和系统。 2. 人力资源&#xff1a;数字化建设需要专业的技术…

K8S - 理解ClusterIP - 集群内部service之间的反向代理和loadbalancer

在Micro Service的治理中。 有两个很重要的点&#xff0c; 集群外部的用户/service 如何访问集群内的 入口服务(例如UI service&#xff09;集群内的service A 如何 访问 集群内的service B 为什么有上面的问题 无非是&#xff1a; 集群内的service 都是多实例的每个servic…

GIM: Learning Generalizable Image Matcher From Internet Videos

【引用格式】&#xff1a;Shen X, Yin W, Mller M, et al. GIM: Learning Generalizable Image Matcher From Internet Videos[C]//The Twelfth International Conference on Learning Representations. 2023. 【网址】&#xff1a;https://arxiv.org/pdf/2402.11095 【开源代…

什么是慢查询——Java全栈知识(26)

1、什么是慢查询 慢查询&#xff1a;也就是接口压测响应时间过长&#xff0c;页面加载时间过长的查询 原因可能如下&#xff1a; 1、聚合查询 2、多表查询 3、单表数据量过大 4、深度分页查询&#xff08;limit&#xff09; 如何定位慢查询&#xff1f; 1、Skywalking 我们…

AIGC系列之一-一文理解什么是Embedding嵌入技术

摘要&#xff1a;嵌入技术&#xff08;Embedding&#xff09;是一种将高维数据映射到低维空间的技术&#xff0c;在人工智能与图形学研究中被广泛应用。本文将介绍嵌入技术的基本概念、原理以及在 AIGC&#xff08;Artificial Intelligence and Graphics Computing&#xff09;…

轻松上手MYSQL:MYSQL事务隔离级别的奇幻之旅

​&#x1f308; 个人主页&#xff1a;danci_ &#x1f525; 系列专栏&#xff1a;《设计模式》《MYSQL》 &#x1f4aa;&#x1f3fb; 制定明确可量化的目标&#xff0c;坚持默默的做事。 ✨欢迎加入探索MYSQL索引数据结构之旅✨ &#x1f44b; 大家好&#xff01;文本学习…

C++封装、继承、多态的应用---职工管理系统

C封装、继承、多态的应用—职工管理系统 文章目录 C封装、继承、多态的应用---职工管理系统1.需求分析2.抽象类的建立2.1抽象基类2.2员工类2.3经理类2.4老板类2.5存储类 3.抽象类的实现4.功能函数的实现4.1菜单功能的实现4.2增加职工功能函数实现4.2显示职工功能函数实现4.3删除…

力扣SQL50 销售分析III having + 条件计数

Problem: 1084. 销售分析III &#x1f468;‍&#x1f3eb; 参考题解 Code select s.product_id,p.product_name from sales s left join product p on s.product_id p.product_id group by product_id having count(if(sale_date between 2019-01-01 and 2019-03-31,1,nu…

【2024最新版】Mysql数据库安装全攻略:图文详解(Windows版本)

目录 1. 引言1.1 MySQL特性1.2 开源1.3 跨平台支持1.4 编程接口1.5 系统特性1.6 性能优势 2. 安装版本选择3. 安装MySQL3.1 下载MySQL3.2 安装MySQL 1. 引言 MySQL是一种流行的关系型数据库管理系统&#xff08;RDBMS&#xff09;&#xff0c;具有高度的可靠性、可扩展性和性能…

C++系列-String(二)

&#x1f308;个人主页&#xff1a;羽晨同学 &#x1f4ab;个人格言:“成为自己未来的主人~” #define _CRT_SECURE_NO_WARNINGS #include<string> #include<iostream> #include<list> #include<algorithm> using namespace std; void test_string…

计算机组成入门知识

前言&#x1f440;~ 数据库的知识点先暂且分享到这&#xff0c;接下来开始接触计算机组成以及计算机网络相关的知识点&#xff0c;这一章先介绍一些基础的计算机组成知识 一台计算机如何组成的&#xff1f; 存储器 CPU cpu的工作流程 主频 如何衡量CPU好坏呢&#xff1f…

基于stm32的温度采集并且显示

目录 一、I2C总线通信协议 &#xff08;一&#xff09;I2C简介 &#xff08;二&#xff09;I2C物理层 &#xff08;三&#xff09;I2C协议层 1、I2C基本读写过程 2、通信的起始和停止信号 3、数据的有效性 4、地址及数据方向 5、响应 &#xff08;四&#xff09;软件I…

常说的云VR是什么意思?与传统vr的区别

虚拟现实&#xff08;Virtual Reality&#xff0c;简称VR&#xff09;是一种利用计算机技术模拟产生一个三维空间的虚拟世界&#xff0c;让用户通过视觉、听觉、触觉等感官&#xff0c;获得与现实世界类似或超越的体验。VR技术发展历程可追溯至上世纪&#xff0c;经历概念提出、…