我有一个属性文件,并使用Spring Property Plote持有人,将值设置为Spring Bean。现在,可以在运行时间内修改此属性文件。有没有一种方法可以用这种新修改的属性值来刷新弹簧豆的属性?尤其是我有很多单身豆吗?如何用新值刷新它们?是否已经有一个解决方案,还是应该自定义编码?如果它尚未存在,请有人可以采取最佳方法来实现这一目标吗?谢谢!

PS:我的应用程序是批处理应用程序。我使用基于弹簧的石英配置来安排批处理。

有帮助吗?

解决方案

我将留下参考,但更新的答案在分隔线以下:

好吧 ConfigurableApplicationContext 界面包含a 刷新() 方法,这应该是您想要的,但是问题是:如何访问该方法。无论您做哪种方式,您都将从具有类型依赖性的bean开始 ConfigurableApplicationContext:

private ConfigurableApplicationContext context;
@Autowired
public void setContext(ConfigurableApplicationContext ctx){
    this.context = ctx;
}

现在我建议的两个基本选择是

  1. 使用 任务执行框架 并让您的Bean定期观看属性资源,在发现更改或
  2. 将豆暴露于JMX, ,允许您手动触发刷新。

参考评论:由于似乎不可能刷新整个上下文,因此另一种策略是创建一个属性工厂bean并将其注入所有其他豆类。

public class PropertiesFactoryBean implements FactoryBean<Properties>{

    public void setPropertiesResource(Resource propertiesResource){
        this.propertiesResource = propertiesResource;
    }

    private Properties value=null;
    long lastChange = -1L;

    private Resource propertiesResource;

    @Override
    public Properties getObject() throws Exception{
        synchronized(this){
            long resourceModification = propertiesResource.lastModified();
            if(resourceModification != lastChange){
                Properties newProps = new Properties();
                InputStream is = propertiesResource.getInputStream();
                try{
                    newProps.load(is);
                } catch(IOException e){
                    throw e;
                } finally{
                    IOUtils.closeQuietly(is);
                }
                value=newProps;
                lastChange= resourceModification;
            }
        }
        // you might want to return a defensive copy here
        return value;
    }

    @Override
    public Class<?> getObjectType(){
        return Properties.class;
    }

    @Override
    public boolean isSingleton(){
        return false;
    }

}

您可以将此属性豆注入所有其他豆类中,但是,您必须谨慎使用原型范围。这在Singleton Bean中特别棘手, 可以在这里找到解决方案.

如果您不想在整个地方注入查找方法,也可以注入 PropertyProvider 这样的豆子:

public class PropertiesProvider implements ApplicationContextAware{

    private String propertyBeanName;
    private ApplicationContext applicationContext;

    public void setPropertyBeanName(final String propertyBeanName){
        this.propertyBeanName = propertyBeanName;
    }

    @Override
    public void setApplicationContext(final ApplicationContext applicationContext) throws BeansException{
        this.applicationContext = applicationContext;
    }

    public String getProperty(final String propertyName){
        return ((Properties) applicationContext.getBean(propertyBeanName)).getProperty(propertyName);
    }

}
许可以下: CC-BY-SA归因
不隶属于 StackOverflow
scroll top