首页 > 解决方案 > Spring Factory 方法“jdbcTemplate”抛出异常;属性“数据源”是必需的

问题描述

我正在尝试使用 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);
    }
}

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

十分感谢

标签: javaspringdatasourcejndijdbctemplate

解决方案


弹簧按类型连接所有 bean。DataSource这意味着当使用 java 配置时,它将寻找具有返回类型的 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;

推荐阅读