org.springframework.dao.InvalidDataAccessApiUsageException:在只读模式下不允许写操作

时间:2014-03-20 06:02:50

标签: java spring hibernate ssh

我是SSH的新手,我使用的是Spring 4,Hibernate 4和Struts 2。 当我为DAO类创建单元测试时,我遇到了这个例外。

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"classpath*:testApplicationContext.xml"})
public class UserDaoTest {
    @Test
    public void testAddUserNotExists() {
        User user = new User("20116524", "785ee107c11dfe36de668b1ae7baacbb");
        userDao.addUser(user);

        assertNotNull(userDao.getUserByUsername("20116524"));
    }

    @Autowired
    public void setUserDao(UserDao userDao) {
        this.userDao = userDao;
    }

    /**
     * UserDao object to test.
     */
    private UserDao userDao;
}

我尝试将以下行添加到web.xml,但它不起作用。

  

参考:Java / Hibernate - Write operations are not allowed in read-only mode

我相信我的applicationContext.xml存在问题,而且内容如下:

<!-- Session Factory -->
<bean id="sessionFactory"
    class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
    <property name="dataSource" ref="dataSource" />
    <property name="configLocation" value="classpath:hibernate.cfg.xml"/>
    <property name="hibernateProperties">
        <props>
            <prop key="hibernate.dialect">org.hibernate.dialect.HSQLDialect</prop>
            <prop key="hibernate.show_sql">true</prop>
        </props>
    </property>
    <property name="packagesToScan">
        <list>
            <value>emis.accounts.model</value>
        </list>
    </property>
</bean>
<!-- Transaction -->
<tx:annotation-driven transaction-manager="transactionManager" />
<bean id="transactionManager"
    class="org.springframework.orm.hibernate4.HibernateTransactionManager">
    <property name="sessionFactory" ref="sessionFactory" />
</bean>

没有关于AOPtx:advice的配置。所以我尝试添加一些行:

<!-- Transaction -->
<bean id="transactionManager"
    class="org.springframework.orm.hibernate4.HibernateTransactionManager">
    <property name="sessionFactory" ref="sessionFactory" />
</bean>

<tx:advice id="txAdvice" transaction-manager="transactionManager">
    <tx:attributes>
        <tx:method name="*" propagation="REQUIRED" read-only="true"/>
        <tx:method name="get*" propagation="REQUIRED" read-only="true" />
        <tx:method name="find*" propagation="REQUIRED" read-only="true" />
        <tx:method name="list*" propagation="REQUIRED" read-only="true" />
        <tx:method name="load*" propagation="REQUIRED" read-only="true" />
        <tx:method name="save*" propagation="REQUIRED" rollback-for="Exception" />
        <tx:method name="add*" propagation="REQUIRED" rollback-for="Exception" />
        <tx:method name="save" propagation="REQUIRED" rollback-for="Exception" />
        <tx:method name="update*" propagation="REQUIRED" rollback-for="Exception" />
        <tx:method name="update" propagation="REQUIRED" rollback-for="Exception" />
        <tx:method name="delete*" propagation="REQUIRED" rollback-for="Exception" />
        <tx:method name="delete" propagation="REQUIRED" rollback-for="Exception" />
        <tx:method name="*" propagation="REQUIRED" />
    </tx:attributes>
</tx:advice>

<aop:aspectj-autoproxy />
<aop:config>
    <aop:pointcut id="aopPointcut"
        expression="execution(* emis.*.service.*.*(..))" />
    <aop:advisor advice-ref="txAdvice" pointcut-ref="aopPointcut"/>
</aop:config>

然而,我有另一个例外:

WARNING: Exception thrown from ApplicationListener handling ContextClosedEvent
java.lang.IllegalStateException: ApplicationEventMulticaster not initialized - call 'refresh' before multicasting events via the context: Root WebApplicationContext: startup date [Thu Mar 20 13:53:51 CST 2014]; root of context hierarchy
    at org.springframework.context.support.AbstractApplicationContext.getApplicationEventMulticaster(AbstractApplicationContext.java:346)
    at org.springframework.context.support.AbstractApplicationContext.publishEvent(AbstractApplicationContext.java:333)
    at org.springframework.context.support.AbstractApplicationContext.doClose(AbstractApplicationContext.java:880)
...

WARNING: Exception thrown from LifecycleProcessor on context close
java.lang.IllegalStateException: LifecycleProcessor not initialized - call 'refresh' before invoking lifecycle methods via the context: Root WebApplicationContext: startup date [Thu Mar 20 13:53:51 CST 2014]; root of context hierarchy
    at org.springframework.context.support.AbstractApplicationContext.getLifecycleProcessor(AbstractApplicationContext.java:359)
    at org.springframework.context.support.AbstractApplicationContext.doClose(AbstractApplicationContext.java:888)
    at org.springframework.context.support.AbstractApplicationContext.close(AbstractApplicationContext.java:841)
...

我该怎么办?谁能帮我?非常感谢!!

1 个答案:

答案 0 :(得分:0)

首先要考虑要用于交易的内容。 XML配置或注释@Transactional。不要试图混合他们两个。

如果您使用@Transactional,请确保使用正确的注释注释UserDao

如果您使用XML,请确保tx:advice内的排序正确无误。目前一切都是只读的。第一场比赛获胜,因为您在顶部有一个通配符*,所有内容都将匹配该切入点。接下来,您不需要<aop:aspectj-autoproxy />,因为您已使用<aop:config />块声明了您的方面。