文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

SpringBoot之注入不了的Spring占位符问题怎么解决

2023-07-05 20:49

关注

本文小编为大家详细介绍“SpringBoot之注入不了的Spring占位符问题怎么解决”,内容详细,步骤清晰,细节处理妥当,希望这篇“SpringBoot之注入不了的Spring占位符问题怎么解决”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。

Spring里的占位符

spring里的占位符通常表现的形式是:

<bean id="dataSource" destroy-method="close" class="org.apache.commons.dbcp.BasicDataSource">    <property name="url" value="${jdbc.url}"/></bean>

或者

@Configuration@ImportResource("classpath:/com/acme/properties-config.xml")public class AppConfig {    @Value("${jdbc.url}")    private String url;}

Spring应用在有时会出现占位符配置没有注入,原因可能是多样的。

本文介绍两种比较复杂的情况。

占位符是在Spring生命周期的什么时候处理的

Spirng在生命周期里关于Bean的处理大概可以分为下面几步:

SpringBoot之注入不了的Spring占位符问题怎么解决

当然这只是比较理想的状态,实际上因为Spring Context在构造时,也需要创建很多内部的Bean,应用在接口实现里也会做自己的各种逻辑,整个流程会非常复杂。

那么占位符(${}表达式)是在什么时候被处理的?

结合上面的Spring的生命周期,如果Bean的创建和使用在PropertySourcesPlaceholderConfigurer之前,那么就有可能出现占位符没有被处理的情况。

例子1

Mybatis 的 MapperScannerConfigurer引起的占位符没有处理

首先应用自己在代码里创建了一个DataSource,其中${db.user}是希望从application.properties里注入的。

代码在运行时会打印出user的实际值。

@Configurationpublic class MyDataSourceConfig {    @Bean(name = "dataSource1")    public DataSource dataSource1(@Value("${db.user}") String user) {        System.err.println("user: " + user);        JdbcDataSource ds = new JdbcDataSource();        ds.setURL("jdbc:h3:˜/test");        ds.setUser(user);        return ds;    }}

然后应用用代码的方式来初始化mybatis相关的配置,依赖上面创建的DataSource对象

@Configurationpublic class MybatisConfig1 {    @Bean(name = "sqlSessionFactory1")    public SqlSessionFactory sqlSessionFactory1(DataSource dataSource1) throws Exception {        SqlSessionFactoryBean sqlSessionFactoryBean = new SqlSessionFactoryBean();        org.apache.ibatis.session.Configuration ibatisConfiguration = new org.apache.ibatis.session.Configuration();        sqlSessionFactoryBean.setConfiguration(ibatisConfiguration);        sqlSessionFactoryBean.setDataSource(dataSource1);        sqlSessionFactoryBean.setTypeAliasesPackage("sample.mybatis.domain");        return sqlSessionFactoryBean.getObject();    }    @Bean    MapperScannerConfigurer mapperScannerConfigurer(SqlSessionFactory sqlSessionFactory1) {        MapperScannerConfigurer mapperScannerConfigurer = new MapperScannerConfigurer();        mapperScannerConfigurer.setSqlSessionFactoryBeanName("sqlSessionFactory1");        mapperScannerConfigurer.setBasePackage("sample.mybatis.mapper");        return mapperScannerConfigurer;    }}

当代码运行时,输出结果是:

user: ${db.user}

为什么会user这个变量没有被注入?

分析下Bean定义,可以发现MapperScannerConfigurer它实现了BeanDefinitionRegistryPostProcessor

这个接口在是Spring扫描Bean定义时会回调的,远早于BeanFactoryPostProcessor

所以原因是:

要解决这个问题,可以在代码里,显式来处理占位符:

environment.resolvePlaceholders("${db.user}")

例子2

Spring boot自身实现问题,导致Bean被提前初始化

Spring Boot里提供了@ConditionalOnBean,这个方便用户在不同条件下来创建bean。

里面提供了判断是否存在bean上有某个注解的功能。

@Target({ ElementType.TYPE, ElementType.METHOD })@Retention(RetentionPolicy.RUNTIME)@Documented@Conditional(OnBeanCondition.class)public @interface ConditionalOnBean {        Class<? extends Annotation>[] annotation() default {};

比如用户自己定义了一个Annotation:

@Target({ ElementType.TYPE })@Retention(RetentionPolicy.RUNTIME)public @interface MyAnnotation {}

然后用下面的写法来创建abc这个bean,意思是当用户显式使用了@MyAnnotation(比如放在main class上),才会创建这个bean。

@Configurationpublic class MyAutoConfiguration {    @Bean    // if comment this line, it will be fine.    @ConditionalOnBean(annotation = { MyAnnotation.class })    public String abc() {        return "abc";    }}

这个功能很好,但是在spring boot 1.4.5 版本之前都有问题,会导致FactoryBean提前初始化。

在例子里,通过xml创建了javaVersion这个bean,想获取到java的版本号。

这里使用的是spring提供的一个调用static函数创建bean的技巧。

<bean id="sysProps" class="org.springframework.beans.factory.config.MethodInvokingFactoryBean">  <property name="targetClass" value="java.lang.System" />  <property name="targetMethod" value="getProperties" /></bean><bean id="javaVersion" class="org.springframework.beans.factory.config.MethodInvokingFactoryBean">  <property name="targetObject" ref="sysProps" />  <property name="targetMethod" value="getProperty" />  <property name="arguments" value="${java.version.key}" /></bean>

我们在代码里获取到这个javaVersion,然后打印出来:

@SpringBootApplication@ImportResource("classpath:/demo.xml")public class DemoApplication {    public static void main(String[] args) {        ConfigurableApplicationContext context = SpringApplication.run(DemoApplication.class, args);        System.err.println(context.getBean("javaVersion"));    }}

在实际运行时,发现javaVersion的值是null。

这个其实是spring boot的锅,要搞清楚这个问题,先要看@ConditionalOnBean的实现。

private String[] getBeanNamesForAnnotation(    ConfigurableListableBeanFactory beanFactory, String type,    ClassLoader classLoader, boolean considerHierarchy) throws LinkageError {  String[] result = NO_BEANS;  try {    @SuppressWarnings("unchecked")    Class<? extends Annotation> typeClass = (Class<? extends Annotation>) ClassUtils        .forName(type, classLoader);    result = beanFactory.getBeanNamesForAnnotation(typeClass);

实现spring boot starter要注意不能导致bean提前初始化

用户在实现spring boot starter时,通常会实现Spring的一些接口,比如BeanFactoryPostProcessor接口,在处理时,要注意不能调用类似beanFactory.getBeansOfTypebeanFactory.getBeanNamesForAnnotation 这些函数,因为会导致一些bean提前初始化。

而上面有提到PropertySourcesPlaceholderConfigurer的order是最低优先级的,所以用户自己实现的BeanFactoryPostProcessor接口在被回调时很有可能占位符还没有被处理。

对于用户自己定义的@ConfigurationProperties对象的注入,可以用类似下面的代码:

@ConfigurationProperties(prefix = "spring.my")public class MyProperties {    String key;}
public static MyProperties buildMyProperties(ConfigurableEnvironment environment) {  MyProperties myProperties = new MyProperties();  if (environment != null) {    MutablePropertySources propertySources = environment.getPropertySources();    new RelaxedDataBinder(myProperties, "spring.my").bind(new PropertySourcesPropertyValues(propertySources));  }  return myProperties;}

读到这里,这篇“SpringBoot之注入不了的Spring占位符问题怎么解决”文章已经介绍完毕,想要掌握这篇文章的知识点还需要大家自己动手实践使用过才能领会,如果想了解更多相关内容的文章,欢迎关注编程网行业资讯频道。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     801人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     348人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     311人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     432人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     220人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯