Spring依赖注入原理分析

Spring中有几种依赖注入的方式?


这是一个面试高频题,但是我在面程序员的时候,听过很多种答案。那么标准答案是什么?我们先不说,一步步来分析。


什么是依赖注入

首先,我们得知道什么是依赖注入?就是填充属性。


一个对象通常都会有属性,比如:


<code>public class OrderService {
    private UserService userService;
    
    public UserService getUserService() {
        return userService;
    }
}/<code>


OrderService中有一个属性UserService, UserService就是OrderService的依赖。


那么Spring的依赖注入,就是Spring框架去进行属性的填充。那么我们就要站在Spring的角度去思考:如果你是Spring的开发者,如果实现对一个对象的属性进行填充?


在进行属性填充之前,我们得先知道:在一个对象中,哪些属性可以进行填充?


肯定是需要业务开发者去告诉程序员的,比如在属性上加一个特定的注解,比如Spring中的@Autowired。当Spring遇到该属性时,发现该属性存在这个注解,Spring就会对当前这个属性进行填充。那么怎么填充呢?


填充其实就是对属性进行赋值,那么Spring能怎么对这个属性进行赋值呢?赋的什么值呢?


假设赋的值是xx。我们先考虑如果把这个xx赋值给OrderService对象。


可能我们立马能想到的就是:

<code>orderService.userService = xx;/<code>


这个思路没错,但是不能满足所有情况,因为orderService这个对象是在Spring中实例化的,userService这个属性的权限修饰符是private,所以在Spring中不能直接进行赋值,但是可以通过反射,比如:


<code>// 随便new一个表示xx,Spring中寻找要注入的值是一个比较复杂的过程
UserService xx = new UserService();

Class c= Class.forName("com.luban.service.OrderService");
Object cInstance = c.newInstance();
Field[] fields = c.getDeclaredFields();
fields[0].setAccessible(true);  // fields[0]表示的就是userServce属性
fields[0].set(cInstance, xx);
System.out.println(((OrderService)cInstance).getUserService());/<code>


这样,通过反射就能对属性进行赋值了。那么怎么寻找到准确的应该赋值给该属性的值呢?


上文中的xx肯定也是一个对象,也就是说是Spring中的一个bean,那么Spring该如何根据当前属性去找到对应的bean呢?只有两种方式:


  1. 根据属性的名字
  2. 根据属性的类型


根据属性的名字去Spring容器中去找bean,要么找不到,要么就能找到一个bean,因为Spring中的beanName是唯一的。


格局属性的类型去Spring容器中去找bean,要么找不到,要么可能找到一个或多个bena,因为Spring容器中的bean实际就是一个对象,而一个类型是可以有多个对象的,在Spring容器中也是如此。


那么Spring针对这两种方式会如何选择呢?二选一,还是二合一。


答案很明显,肯定是二合一,一种方式找不到就利用另外一个方式去找。那么两种方式中会优先利用哪种方式去找?


答案很明显,肯定是先利用属性的名字,因为利用名字去找更精确。

后续还有一部分没写完,之后会更新的

Spring依赖注入原理分析


分享到:


相關文章: