【问题标题】:Spring Factory method 'jdbcTemplate' threw exception; Property 'dataSource' is requiredSpring Factory 方法“jdbcTemplate”抛出异常;属性“数据源”是必需的
【发布时间】:2021-05-23 12:03:22
【问题描述】:

我正在尝试使用 Spring 4 和数据源配置连接到数据库。 我正在学习教程,所以这是正确的配置:

package spittr.config;

import ...

@Configuration
public class DataConfig {

    @Bean
    public JndiObjectFactoryBean dataSource() {
        JndiObjectFactoryBean jndiObjectFB = new JndiObjectFactoryBean();
        jndiObjectFB.setJndiName("java:jboss/datasources/jdbc/SpitterDS");
        jndiObjectFB.setProxyInterface(javax.sql.DataSource.class);
        return jndiObjectFB;
    }

    @Bean
    public JdbcOperations jdbcTemplate(DataSource dataSource) {
        return new JdbcTemplate(dataSource);
    }
}

一切正常,但如您所见,我返回的是 JndiObjectFactoryBean 而不是 Datasource。如果我对 Spring 理解得很好(可能我不理解,否则在这里我会理解),如果你不指定 Bean 名称,Spring 会将 bean 的名称设置为返回类型,第一个字母为小写。 例如,以下代码行将返回一个 id 为“myFantasticBean”的bean(“m”小写)

@Bean
public MyFantasticBean createMyBean() {
    return new MyFantasticBean();
}

我在网上看到很多人使用这个版本的 DataConfig,其中方法 dataSource() 返回一个 DataSource 类型的对象(应该是这样):

package spittr.config;

import ...

@Configuration
public class DataConfig {

    @Bean
    public DataSource dataSource() {
        JndiObjectFactoryBean jndiObjectFB = new JndiObjectFactoryBean();
        jndiObjectFB.setJndiName("java:jboss/datasources/jdbc/SpitterDS");
        jndiObjectFB.setProxyInterface(javax.sql.DataSource.class);
        return (DataSource) jndiObjectFB.getObject();
    }

    @Bean
    public JdbcOperations jdbcTemplate(DataSource dataSource) {
        return new JdbcTemplate(dataSource);
    }
}

但是如果我使用这个 DataSource 创建,我会得到以下错误:

bin/content/10_SpringWeb_BE_JDBC.war/WEB-INF/classes/spittr/data/JdbcSpitterRepository.class"]: Unsatisfied dependency expressed through constructor parameter 0; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'jdbcTemplate' defined in class path resource [spittr/config/DataConfig.class]: Bean instantia
tion via factory method failed; nested exception is org.springframework.beans.BeanInstantiationException: Failed to instantiate [org.springframework.jdbc.core.JdbcOperations]: Factory method 'jdbcTemplate' threw exception; nested exception is java.lang.IllegalArgumentException: Property 'dataSource' is required

我什至找到了解决方案,修改了 dataSource() 方法,现在变成了这样:

package spittr.config;

import ...

@Configuration
public class DataConfig {

    @Bean
    public DataSource dataSource(){
        final JndiDataSourceLookup dsLookup = new JndiDataSourceLookup();
        DataSource dataSource = dsLookup.getDataSource("java:jboss/datasources/jdbc/SpitterDS");
        return dataSource;
    }

    @Bean
    public JdbcOperations jdbcTemplate(DataSource dataSource) {
        return new JdbcTemplate(dataSource);
    }
}

但我真的不明白为什么这行得通,而以前的行不通。有人可以向我解释我做错了什么吗?

非常感谢

【问题讨论】:

    标签: java spring datasource jndi jdbctemplate


    【解决方案1】:

    Spring 按类型连接所有 bean。这意味着当使用 java 配置时,它将查找返回类型为 DataSource 的 bean 定义。这是您在使用 JndiDataSourceLookup 时所拥有的,因此 Spring 可以毫无问题地找到它。 它不能从 JndiObjectFactoryBean 派生 DataSource,除非您在返回它时强制转换它。

    @Bean
    public DataSource dataSource() {
       JndiObjectFactoryBean jndiObjectFB = new JndiObjectFactoryBean();     
       jndiObjectFB.setJndiName("java:jboss/datasources/jdbc/SpitterDS");
       jndiObjectFB.setProxyInterface(javax.sql.DataSource.class);
       return (DataSource) jndiObjectFB.getObject();
    }
    

    另外,你说的有点不对

    如果我对 Spring 有很好的理解(可能我没有,否则我会 到这里就明白了),如果不指定Bean名称,Spring会 将 bean 的名称设置为第一个返回的类型 小写字母。例如,以下代码行将 返回一个 id 为“myFantasticBean”的bean(“m”小写)

    仅在使用组件扫描时才适用。

    所以如果你有一个用例如注释的类@Service 这样的:

    @Service
    public class MyFantasticBean { .. }
    

    那么是的,名称将是 myFantasticBean,因为该名称不能从其他东西派生。 在使用 Java 配置时,重要的是要意识到名称与类型不同。

    @Bean
    public MyFantasticBean createMyBean() {
        return new MyFantasticBean();
    }
    

    在您的示例中,MyFantasticBean 是当您想要注入它时 spring 将搜索的 bean 类型。 createMyBean 将是 bean 的名称。因此,如果您有多个 MyFantasticBean 类型的 bean 实例,那么您可以使用 bean 名称来指定要注入的 bean。在你的情况下,那将是

    @Qualifier("createMyBean")
    @Autowired
    private MyFantasticBean myFantasticBean;
    

    【讨论】:

    • 嗨,尼科,感谢您的帮助。我有两个问题:1)如您所见,我将返回值转换为(DataSource),那么为什么会出现错误? 2)我正在从“Spring4inAction”一书中学习。我在下面写了一个关于 bean 命名的示例:@Bean public CompactDisc sgtPeppers(){ return new SgtPeppers(); } 这本书说:“默认情况下,bean 将被赋予一个与 @Bean 注释方法的名称相同的 ID。在这种情况下,bean 将被命名压缩碟片。”这是错的吗? bean 是否应该命名为 sgtPeppers(方法名)而不是 compactDisc(result Type)?
    • 1) 我不太确定。很难像这样说为什么工厂无法创建数据源。 2)是的,不幸的是,这本书(一半)不正确。 bean 名称确实是从方法名称派生的,但正如您所猜测的 sgtPeppers。
    • 顺便说一句,您可以使用 applicationContext.getBeanDefinitionNames() 获取所有 bean 名称。
    • @PyroSandro 如果这对您有任何帮助,您能接受我的回答吗?谢谢
    猜你喜欢
    • 2014-10-30
    • 1970-01-01
    • 1970-01-01
    • 2011-01-10
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2011-07-03
    • 1970-01-01
    相关资源
    最近更新 更多