【问题标题】:Spring 3 JavaConfig - Problems Injecting DataSourceSpring 3 JavaConfig - 注入数据源的问题
【发布时间】:2013-12-06 01:58:38
【问题描述】:

我是 Spring 新手,我正在尝试让我的 DataSource 注入工作。我收到“需要属性‘数据源’”错误。被这件事难住了一段时间。提前感谢您的帮助!

WebConfig.java

import javax.sql.DataSource;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.EnableWebMvc;
import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter;
import org.springframework.web.servlet.view.InternalResourceViewResolver;

@Configuration
@EnableWebMvc
@ComponentScan(basePackages = "com.test")
public class WebConfig extends WebMvcConfigurerAdapter {

@Autowired
DataSource Source;

@Override
public void addResourceHandlers(ResourceHandlerRegistry registry) {
    System.out.println("addResourceHandlers :: init");
    registry.addResourceHandler("/resources/**").addResourceLocations(
            "/resources/");
}

@Bean
public InternalResourceViewResolver viewResolver() {
    System.out.println("viewResolver :: init");
    InternalResourceViewResolver resolver = new InternalResourceViewResolver();
    resolver.setPrefix("/WEB-INF/views/");
    resolver.setSuffix(".jsp");
    return resolver;
}

}

AppConfig.java

import javax.sql.DataSource;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.jdbc.datasource.DriverManagerDataSource;

@Configuration
@ComponentScan(basePackages = { "com.test" })
public class AppConfig {

@Bean
public DataSource dataSource() {
    System.out.println("userDBDatasource :: init");
    DriverManagerDataSource dataSource = new DriverManagerDataSource();
    dataSource.setDriverClassName("com.mysql.jdbc.Driver");
    dataSource.setUrl("jdbc:mysql://localhost:3306/test");
    dataSource.setUsername("foo");
    dataSource.setPassword("foo");
    return dataSource;
}

}

JDBCUserDaoImpl.java

package com.test.foo;

import java.util.ArrayList;
import java.util.List;
import javax.sql.DataSource;
import org.apache.log4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Repository;

@Repository
public class JDBCUserDAOImpl implements JDBCUserDAO {

private Logger logger = Logger.getRootLogger();
private JdbcTemplate jdbcTemplate;
private DataSource dataSource;

@Autowired
public void setDataSource(DataSource dataSource) {
    logger.debug("setDataSource :: called");
    this.dataSource = dataSource;
}

public List<UserBean> getUsers() {
    logger.info("Getting List of Users!");
    List<UserBean> userBeans = new ArrayList<UserBean>();
    String sql = "SELECT * FROM users";
    jdbcTemplate = new JdbcTemplate(dataSource);
    userBeans = jdbcTemplate.query(sql, new UserBeanRowMapper());
    return userBeans;
}
}

错误

Dec 07, 2013 10:46:53 AM org.apache.catalina.core.StandardWrapperValve invoke
SEVERE: Servlet.service() for servlet [main] in context with path [/springmvc] threw exception [Request processing failed; nested exception is java.lang.IllegalArgumentException: Property 'dataSource' is required] with root cause
java.lang.IllegalArgumentException: Property 'dataSource' is required
    at org.springframework.jdbc.support.JdbcAccessor.afterPropertiesSet(JdbcAccessor.java:134)
    at org.springframework.jdbc.core.JdbcTemplate.<init>(JdbcTemplate.java:165)
    at com.tajima.models.JDBCUserDAOImpl.getUsers(JDBCUserDAOImpl.java:42)
    at com.tajima.controllers.HomeController.loadUserList(HomeController.java:53)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:606)
    at org.springframework.web.method.support.InvocableHandlerMethod.invoke(InvocableHandlerMethod.java:219)
    at org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:132)
    at org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:104)
    at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandleMethod(RequestMappingHandlerAdapter.java:745)
    at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:686)
    at org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:80)
    at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:925)
    at org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:856)
    at org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:936)
    at org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:827)
    at javax.servlet.http.HttpServlet.service(HttpServlet.java:621)
    at org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:812)
    at javax.servlet.http.HttpServlet.service(HttpServlet.java:728)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:305)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:210)
    at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:222)
    at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:123)
    at org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:502)
    at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:171)
    at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:99)
    at org.apache.catalina.valves.AccessLogValve.invoke(AccessLogValve.java:953)
    at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:118)
    at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:408)
    at org.apache.coyote.http11.AbstractHttp11Processor.process(AbstractHttp11Processor.java:1023)
    at org.apache.coyote.AbstractProtocol$AbstractConnectionHandler.process(AbstractProtocol.java:589)
    at org.apache.tomcat.util.net.JIoEndpoint$SocketProcessor.run(JIoEndpoint.java:312)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1145)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:615)
    at java.lang.Thread.run(Thread.java:744)

【问题讨论】:

  • 你能发布完整的错误吗?
  • 以及如何加载这些配置文件。关于 DAAO 的性能提示,您不应该在每次需要时都创建 JdbcTemplate,构建一次然后重复使用。
  • 感谢您的提示!我已经添加了错误。
  • 向我们展示如何在控制器中获取 DAO 实例。我的猜测是您使用new JDBCUserDAOImpl() 来获取一个,而不是使用依赖注入。
  • 对不起,也许我还没有很好地理解这一点。我认为在我的 setDataSource 方法上使用 autowired 注释会使 Spring 注入我的 DataSource bean。

标签: java spring datasource code-injection


【解决方案1】:

@Import 注解提供了这种支持,它直接等同于 Spring beans XML 文件中的元素。

您错过了将 AppConfig 类导入 WebMvcConfig 类文件。因此,spring 无法创建名为“dataSource”的 bean。

在您的 WebMvcConfig 类中,您需要使用 @Import 注释导入 DataSource 配置类。这将有助于通过 spring Container 创建 dataSource bean。

 ex:@Configuration
    @EnableWebMvc
    @Import({AppConfig.class})
    @ComponentScan(basePackages = "com.test")
    public class WebConfig extends WebMvcConfigurerAdapter {

我希望正确地将 webConfig 类指定为 Dispatcher servlet 的 contextConfigLocation 参数。如果不存在,请按以下说明指定

<servlet>
<servlet-name>mvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
  <param-name>contextClass</param-name>
  <param-value>org.springframework.web.context.support.AnnotationConfigWebApplicationContext</param-value>
</init-param>
<init-param>
  <param-name>contextConfigLocation</param-name>
  <param-value>yourpackage.config.WebMvcConfig</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
 <servlet-name>mvc</servlet-name>
 <url-pattern>/</url-pattern>
</servlet-mapping>

或者您可以将这个上下文类和 contextConfiglocation 指定给 ContextLoaderLister 作为上下文参数,如下所示

   <listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
 <context-param>
<param-name>contextClass</param-name>
<param- value>org.springframework.web.context.support.AnnotationConfigWebApplicationC     ontext</param-value>
</context-param>
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>yourpackage.WebMvcConfig</param-value>
</context-param>

【讨论】:

    【解决方案2】:

    从 WebConfig.java 中移除 DataSource 注入

    【讨论】:

      猜你喜欢
      • 2014-06-06
      • 1970-01-01
      • 2014-09-02
      • 2013-02-03
      • 2014-10-21
      • 2018-01-27
      • 1970-01-01
      • 2013-02-09
      • 1970-01-01
      相关资源
      最近更新 更多