1

我对 Spring 和 Java 世界很陌生。我正在尝试编写一个 Web 应用程序来查看示例中提供的宠物诊所应用程序。但是,我从昨天开始就遇到了这个障碍。如果有人能指出我正确的方向,那就太好了。我在谷歌上查找了许多链接,但没有一个解决方案对我有用。

这是错误的根本原因

org.springframework.beans.factory.NoSuchBeanDefinitionException: No matching bean of type [org.hibernate.SessionFactory] found for dependency: expected at least 1 bean which qualifies as autowire candidate for this dependency. Dependency annotations: {}
org.springframework.beans.factory.support.DefaultListableBeanFactory.raiseNoSuchBeanDefinitionException(DefaultListableBeanFactory.java:949)
org.springframework.beans.factory.support.DefaultListableBeanFactory.doResolveDependency(DefaultListableBeanFactory.java:818)
org.springframework.beans.factory.support.DefaultListableBeanFactory.resolveDependency(DefaultListableBeanFactory.java:730)
org.springframework.beans.factory.support.ConstructorResolver.resolveAutowiredArgument(ConstructorResolver.java:795)
org.springframework.beans.factory.support.ConstructorResolver.createArgumentArray(ConstructorResolver.java:723)
org.springframework.beans.factory.support.ConstructorResolver.autowireConstructor(ConstructorResolver.java:196)
org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.autowireConstructor(AbstractAutowireCapableBeanFactory.java:1049)
org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBeanInstance(AbstractAutowireCapableBeanFactory.java:953)
org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:490)

. . .

这是我在 WEB-INF 中的 web.xml

<display-name>Game's List</display-name>

<context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>/WEB-INF/spring/applicationContext-hibernate.xml</param-value>
</context-param>

<session-config>
    <session-timeout>10</session-timeout>
</session-config>

<servlet>
    <servlet-name>gamelist</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <load-on-startup>1</load-on-startup>
</servlet>

<servlet-mapping>
    <servlet-name>gamelist</servlet-name>
    <url-pattern>/</url-pattern>
</servlet-mapping>

这是 applicationContext-hibernate.xml

<!-- ========================= RESOURCE DEFINITIONS ========================= -->

<!-- import the dataSource definition -->
<import resource="applicationContext-dataSource.xml"/>
<!-- Configurer that replaces ${...} placeholders with values from a properties file -->
<!-- (in this case, Hibernate-related settings for the sessionFactory definition below) -->
<context:property-placeholder location="classpath:jdbc.properties"/>
<context:annotation-config />
<!-- Hibernate SessionFactory -->
<bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean"
      p:dataSource-ref="dataSource" p:mappingResources="gamelist-hibernate.xml">
    <property name="hibernateProperties">
        <props>
            <prop key="hibernate.dialect">${hibernate.dialect}</prop>
            <prop key="hibernate.show_sql">${hibernate.show_sql}</prop>
            <prop key="hibernate.generate_statistics">${hibernate.generate_statistics}</prop>
        </props>
    </property>
    <property name="eventListeners">
        <map>
            <entry key="merge">
                <bean class="org.springframework.orm.hibernate3.support.IdTransferringMergeEventListener"/>
            </entry>
        </map>
    </property>
</bean>

<!-- central data access object: Hibernate implementation -->
<bean id="gameDataStore" class="com.gamelist.datastore.GameItemDataStore"/>

<!-- Transaction manager for a single Hibernate SessionFactory (alternative to JTA) -->
<bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager"
      p:sessionFactory-ref="sessionFactory"/>

<!-- ========================= BUSINESS OBJECT DEFINITIONS ========================= -->

<!--
    Activates various annotations to be detected in bean classes:
    Spring's @Required and @Autowired, as well as JSR 250's @Resource.
-->
<context:annotation-config/>

<!--
    Instruct Spring to perform declarative transaction management
    automatically on annotated classes.
-->
<tx:annotation-driven/>
<!--
    Exporter that exposes the Hibernate statistics service via JMX. Autodetects the
    service MBean, using its bean name as JMX object name.
-->
<context:mbean-export/>

这是 gamelist-servlet.xml

<context:component-scan base-package="com.gamelist"/>

<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
    <property name="prefix" value="/WEB-INF/pages/"/>
    <property name="suffix" value=".jsp"/>
</bean>

这是控制器类

    public class GameLibraryController {

    private IDataStore gameDataStore;

    @Autowired
    public GameLibraryController(GameItemDataStore gameDataStore){
        this.gameDataStore = gameDataStore;
    }

    @RequestMapping("/addGame")
    public String addNewGame(Model model){
        model.addAttribute("gameItem",new GameItem());
        return "library/addgameform";
    }

    @RequestMapping("/addGameSubmit")
    public void add(@ModelAttribute GameItem gameItem,BindingResult result, SessionStatus status){
        this.gameDataStore.add(gameItem);
        status.setComplete();
    }
}

这是hibernate实现类GameItemDataStore

  @Repository
    @Transactional
    public class GameItemDataStore implements IDataStore{
    private SessionFactory sessionFactory;

    @Autowired
    public GameItemDataStore(SessionFactory sessionFactory) {
        this.sessionFactory = sessionFactory;
    }

    public void add(BaseItem newGame){
        sessionFactory.getCurrentSession().save(newGame);
    }
}

这是 IDataStore 接口

 public interface IDataStore {

    public void add(BaseItem newGame);
}
4

2 回答 2

6

为避免任何混淆,我将假设您在问题中提到的配置,并从那里开始。(忽略您在阅读有关您的问题的评论后可能尝试过的更改)

先说一点背景:

  1. Spring MVC 应用程序中有两个上下文在起作用。applicationContext更准确地称为,通过与 Servlet 容器的生命周期实现的集成,它可用于所有级别的 spring上下文ROOT WebApplicationContextServlet第二种类型的上下文是为每个DispatcherServlet配置加载的web.xml让我们调用它ServletApplicationContext

  2. 现在,当容器启动时,将加载 Root 应用程序上下文

  3. 加载 Servlet 时,将加载 Servlet 应用程序上下文。此时,来自根应用程序上下文的 bean 可用于 Servlet 应用程序上下文。

  4. 如果,您最终会在两个上下文中加载相同的 bean 定义,就像在 Servlet 上下文中覆盖 bean。加载重复的 bean 效率非常低,如果您依赖于在所有上下文中都可用的同一个实例,可能会导致奇怪的错误。但是,对于像单个 servlet 部署这样的简单情况,这不应该导致您的应用程序在启动时中断,spring 会处理这个问题。

现在了解您的配置有什么问题:

  1. ContextLoaderListener您的web.xml. 如果没有 ContextLoaderListener,则不会加载 ROOT 应用程序上下文。所以没有/WEB-INF/spring/applicationContext-hibernate.xml加载任何bean。

  2. 当您的 servlet 启动时,它会从 加载 bean gamelist-servlet.xml,但无法找到sessionFactorybean,因为从未加载过 ROOT Web 应用程序上下文~!

  3. 为了确认这一点,对于测试,注释掉两个上下文文件中的所有 bean 定义,只<beans></beans>在其中保留空标签。启动你的服务器,你会在最后看到一条日志语句,上面写着类似no ROOT Web Application Context found, configure ContextLoaderListener in web.xml

和修复:

  1. 添加ContextLoaderListener到您的web.xml

  2. 您可以选择像现在一样继续您的配置,或者遵循前面评论中给出的建议,在 servlet 上下文中仅包含与 MVC 相关的定义,而在 ROOT 应用程序上下文中包含所有其他定义。

  3. 另一个更简单的选择是简单地为 ROOT Web 应用程序上下文提供一个空的应用程序上下文 xml,并在 servlet 上下文中配置所有 bean。这从设计的角度来看是不正确的purist,但这是一个实用的简单解决方案,前提是您计划拥有一个 servlet 应用程序。

对于冗长的描述感到抱歉,但我觉得有必要进行一些解释。希望这可以帮助。

于 2013-07-30T12:56:09.813 回答
1

摆脱

<bean id="gameDataStore" class="com.gamelist.datastore.GameItemDataStore"/>

你不需要,因为@Repository你的班级和

<context:annotation-config />

你有两次。然后加

<context:component-scan base-package="com.your.package" /> 

wherecom.your.package是包含该类的顶级包GameItemDataStore

于 2013-07-26T18:54:48.927 回答