Spring 依赖注入 static 静态变量相关问题

1.Spring 不支持依赖注入 static 静态变量

在 springframework 里, 我们不能 @Autowired 一个静态变量, 使之成为一个 spring bean, 例如下面这样:

@Autowired
private static YourClass yourClass;

可以试一下, yourClass 在这种状态下不能够被依赖注入, 会抛出运行时异常 java.lang.NullPointerException, 为什么呢? 静态变量 / 类变量不是对象的属性, 而是一个类的属性, spring 则是基于对象层面上的依赖注入.
而使用静态变量 / 类变量扩大了静态方法的使用范围. 静态方法在 spring 是不推荐使用的. 依赖注入的主要目的, 是让容器去产生一个对象的实例, 然后在整个生命周期中使用他们, 同时也让 testing 工作更加容易.
一旦你使用静态方法, 就不再需要去产生这个类的实例, 这会让 testing 变得更加困难, 同时你也不能为一个给定的类, 依靠注入方式去产生多个具有不同的依赖环境的实例. 这种 static field 是隐含共享的, 并且是一种 global 全局状态, spring 同样不推荐这样去做.

2.Spring 如何给静态变量注入值

spring 不允许 / 不支持把值注入到静态变量中,如:

import org.springframework.beans.factory.annotation.Value;  
import org.springframework.stereotype.Component;  
  
@Component  
public class GlobalValue {  
  
    @Value("${mongodb.db}")  
    public static String DATABASE;  
  
}

如果你获取 GlobalValue.DATABASE,会得到 null
GlobalValue.DATABASE = null

那我们如何解决这个问题呢。
好在 spring 支持 set 方法注入,我们可以利用非静态 setter 方法注入静态变量。如:

import org.springframework.beans.factory.annotation.Value;  
import org.springframework.stereotype.Component;  
  
@Component  
public class GlobalValue {  
  
    public static String DATABASE;  
  
    @Value("${mongodb.db}")  
    public void setDatabase(String db) {  
        DATABASE = db;  
    }  
  
}

输出:
GlobalValue.DATABASE = "mongodb database name"

3.Spring 静态注入的三种方式

Spring 静态注入的三种方式:
(说明:MongoFileOperationUtil 是自己封装的一个 Mongodb 文件读写工具类,里面需要依赖 AdvancedDatastore 对象实例,dsForRW 用来获取 Mongodb 数据源)
在 springframework 里,我们不能 @Autowired 一个静态变量, 使之成为一个 spring bean,例如下面这种方式:

@Autowired  
private static AdvancedDatastore dsForRW;

可以试一下,dsForRW 在这种状态下不能够被依赖注入,会抛出运行时异常 java.lang.NullPointerException,为什么呢? 静态变量 / 类变量不是对象的属性, 而是一个类的属性, spring 则是基于对象层面上的依赖注入。
但是自己比较喜欢封装工具类,并通过 @Component 注解成功能组件,但是功能组件中的方法一般都是静态方法,静态方法只能调用静态成员变量,于是就有了下面的问题。封有的时候封装功能组件会需要底层的 service 注入,怎么办呢?
去网上搜了下解决办法,简单总结一下几种实现方式;
1.xml 方式实现:

<bean init-method="init">  
    <property name="dsForRW" ref="dsForRW"/>  
</bean>
public class MongoFileOperationUtil {  
      
    private static AdvancedDatastore dsForRW;  
  
    private static MongoFileOperationUtil mongoFileOperationUtil;  
  
    public void init() {  
        mongoFileOperationUtil = this;  
        mongoFileOperationUtil.dsForRW = this.dsForRW;  
    }  
  
}

这种方式适合基于 XML 配置的 WEB 项目;
2.@PostConstruct 方式实现;

import org.mongodb.morphia.AdvancedDatastore;  
import org.springframework.beans.factory.annotation.Autowired;  
  
  
@Component  
public class MongoFileOperationUtil {  
    @Autowired  
    private static AdvancedDatastore dsForRW;  
  
    private static MongoFileOperationUtil mongoFileOperationUtil;  
  
    @PostConstruct  
    public void init() {  
        mongoFileOperationUtil = this;  
        mongoFileOperationUtil.dsForRW = this.dsForRW;  
    }  
  
}

@PostConstruct 注解的方法在加载类的构造函数之后执行,也就是在加载了构造函数之后,执行 init 方法;(@PreDestroy 注解定义容器销毁之前的所做的操作)
这种方式和在 xml 中配置 init-method 和 destory-method 方法差不多,定义 spring 容器在初始化 bean 和容器销毁之前的所做的操作;
3.set 方法上添加 @Autowired 注解,类定义上添加 @Component 注解;

import org.mongodb.morphia.AdvancedDatastore;  
import org.springframework.beans.factory.annotation.Autowired;  
import org.springframework.stereotype.Component;  
  
@Component  
public class MongoFileOperationUtil {  
  
    private static AdvancedDatastore dsForRW;  
      
    @Autowired  
    public void setDatastore(AdvancedDatastore dsForRW) {  
        MongoFileOperationUtil.dsForRW = dsForRW;  
    }  
}

首先 Spring 要能扫描到 AdvancedDatastore 的 bean,然后通过 setter 方法注入;
然后注意:成员变量上不需要再添加 @Autowired 注解;