Spring 注入static属性值方式

Spring 注入static属性值

本文介绍Spring中如何从属性文件给static字段注入值。实际应用中一些工具类中static属性值需读取配置文件,实现该功能可以让工具类提供静态方法更易使用。

1. 问题

首先在属性文件中定义属性:

name = Inject a value to a static field

然后给实例变量注入值,通常在字段上使用@Value注解:

@Value("${name}")
private String name;

但在static字段上应用是,会发现其值为null,注入失败:

@Value("${name}")
private static String NAME_NULL;

这是因为Spring不支持在static字段上使用@Value注解。

2. 解决方案

Spring @Value注解可以在方法上使用,在加载所有Spring配置和bean后,Spring上下文将调用它。方法有多个参数,那么每个参数值都为方法注解对应的值,如果需要参数获取不同的值,可以在参数上增加注解:

@Value("Test")
public void printValues(String s, String v){} //both 's' and 'v' values will be 'Test' 
@Value("Test")
public void printValues(String s, @Value("Data") String v){}
// s=Test, v=Data

有了上面的知识,我们可以修改代码为:

public class PropertyUtils { 
    @Value("${name1}")
    private String name; 
    private static String NAME_STATIC; 
    @Value("${name2}")
    public void setNameStatic(String name){
        PropertyController.NAME_STATIC = name;
    }
}

这回通过方法成功给static变量NAME_STATIC赋值。

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静态注入的三种方式

(说明:MongoFileOperationUtil是自己封装的一个Mongodb文件读写工具类,里面需要依赖AdvancedDatastore对象实例,dsForRW用来获取Mongodb数据源)

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

@Autowired  
private static AdvancedDatastore dsForRW;  

可以试一下,dsForRW在这种状态下不能够被依赖注入,会抛出运行时异常java.lang.NullPointerException,为什么呢?静态变量/类变量不是对象的属性,而是一个类的属性,spring则是基于对象层面上的依赖注入。

但是自己比较喜欢封装工具类,并通过@Component注解成功能组件,但是功能组件中的方法一般都是静态方法,静态方法只能调用静态成员变量,于是就有了下面的问题。封有的时候封装功能组件会需要底层的service注入,怎么办呢?

去网上搜了下解决办法,简单总结一下几种实现方式

1.xml方式实现

  
      
 
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注解;

你可能感兴趣的