我的java项目中spring事务总是出问题?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了我的java项目中spring事务总是出问题?相关的知识,希望对你有一定的参考价值。

我的项目用的是spring 4.1.6 hibernate 4.3.8 我的SessionFactory是这么配置的: <bean id="sessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean"> <property name="dataSource" ref="dataSource" /> <property name="hibernateProperties"> <props> <prop key="hibernate.jdbc.batch_size">$hibernate.jdbc.batch_size</prop> <prop key="current_session_context_class">org.springframework.orm.hibernate4.SpringSessionContext</prop> <prop key="hibernate.query.factory_class">org.hibernate.hql.internal.ast.ASTQueryTranslatorFactory</prop> <prop key="hibernate.cache.use_query_cache">$hibernate.cache.use_query_cache</prop> <prop key="hibernate.cache.use_second_level_cache">$hibernate.cache.use_second_level_cache</prop> <prop key="hibernate.cache.region.factory_class">org.hibernate.cache.ehcache.EhCacheRegionFactory</prop> </props> </property> <property name="mappingLocations"> <list> <value>classpath*:com/iresearch/demo/entity/*.hbm.xml</value> </list> </property> </bean>这个是TransactionManager的配置 <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="get*" read-only="true" /> <tx:method name="count*" read-only="true" /> <tx:method name="find*" read-only="true" /> <tx:method name="list*" read-only="true" /> <tx:method name="*" read-only="false" propagation="REQUIRED" /> </tx:attributes> </tx:advice> <aop:config> <aop:pointcut id="txPointcut" expression="execution(* com.iresearch..*.*(..))" /> <aop:advisor pointcut-ref="txPointcut" advice-ref="txAdvice" /> </aop:config>我的包结构是这样的com.iresearch.cloud.entity/dao/service/controller dao层是数据交互层 service是业务层对dao层进行调用我的dao中多数用了SessionFactory的getCurrentSession方法获取session 以及用了hibernateTemplate 在进行单元测试都正常 一旦进入web环境 就会出现org.hibernate.HibernateException: Could not obtain transaction-synchronized Session for current thread的异常 我在spring mvc的配置文件中加入拦截器OpenSessionInViewInterceptor这个问题就不出现了 但会出现org.springframework.dao.InvalidDataAccessApiUsageException: Write operations are not allowed in read-only mode (FlushMode.MANUAL): Turn your Session into FlushMode.COMMIT/AUTO or remove 'readOnly' marker from transaction definition.这个问题看起来我的问题是方法没有加上事务造成session的flushmode全是manual 但是我事务配的没错啊 我换了好几种事务配置方式就是execution表达式换了好几个 都是这样的问题怎么回事啊?

最近在项目的时候碰到pring事务不起作用的情况,后来解决了,这里我汇总下:

1、首先使用如下代码 确认你的bean 是代理对象吗?
必须是Spring定义(通过XML或注解定义都可以)的Bean才接受事务。
直接new出来的对象添加事务是不起作用的。
可以通过以下方式判断是否是代理对象:

AopUtils.isAopProxy(Object object)
AopUtils.isCglibProxy(Object object) //cglib
AopUtils.isJdkDynamicProxy(Object object) //jdk动态代理

2、入口函数必须是public,否则事务不起作用。这一点由Spring的AOP特性决定的。

3、切入点配置错误。
<!-- 使用annotation注解方式配置事务 -->
<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/>
<!--使用切面方式配置事务-->
<tx:advice id="txadvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="*" isolation="READ_COMMITTED" propagation="REQUIRED" rollback-for="java.lang.Exception" />
</tx:attributes>
</tx:advice>

<aop:config>
<aop:pointcut id="pointcut" expression="execution(* com.tyyd..*Service.do*(..))"/>
<aop:advisor advice-ref="txadvice" pointcut-ref="pointcut"/>
</aop:config>

4、如果你使用了springmvc,可能是context:component-scan重复扫描引起的:

5、如使用mysql且引擎是MyISAM造成的(因为不支持事务),改成InnoDB即可。追问

我看了一下 5肯定不是问题原因 都是innoDB 4我只在spring mvc里扫描注解 不存在重复扫描 3貌似我配置没错误 2没看懂 我所有dao和service的方法都是public的 1三个方法参数传入service对象全是false

参考技术A

1、首先使用如下代码 确认你的bean 是代理对象吗?

必须是Spring定义(通过XML或注解定义都可以)的Bean才接受事务。

直接new出来的对象添加事务是不起作用的。

可以通过以下方式判断是否是代理对象:

AopUtils.isAopProxy(Object object)

AopUtils.isCglibProxy(Object object) //cglib

AopUtils.isJdkDynamicProxy(Object object) //jdk动态代理

2、入口函数必须是public,否则事务不起作用。这一点由Spring的AOP特性决定的。

3、切入点配置错误。

<!-- 使用annotation注解方式配置事务 -->

<tx:annotation-driven transaction-manager="transactionManager" proxy-target-class="true"/>

<!--使用切面方式配置事务-->

<tx:advice id="txadvice" transaction-manager="transactionManager">

<tx:attributes>

<tx:method name="*" isolation="READ_COMMITTED" propagation="REQUIRED" rollback-for="java.lang.Exception" />

</tx:attributes>

</tx:advice>

<aop:config>

<aop:pointcut id="pointcut" expression="execution(* com.tyyd..*Service.do*(..))"/>

<aop:advisor advice-ref="txadvice" pointcut-ref="pointcut"/>

</aop:config>

4、如果你使用了springmvc,可能是context:component-scan重复扫描引起的:

5、如使用mysql且引擎是MyISAM造成的(因为不支持事务),改成InnoDB即可。

org.springframework.transaction.UnexpectedRollbackException: Transaction rolled back because it has been marked as rollback-only

检查方法都是查询都没有异常,搜索百度了很多资料还是不解。

java - 如何在java桌面应用程序中使用spring(事务)和hibernate创建嵌入式H2 DB?

【中文标题】java - 如何在java桌面应用程序中使用spring(事务)和hibernate创建嵌入式H2 DB?【英文标题】:How to create emdedded H2 DB with spring(transactional) and hibernate in java desktop application? 【发布时间】:2016-10-26 11:25:15 【问题描述】:

我正在尝试创建一个带有嵌入式 h2 db 的项目,并使用带有 hibernate 的 spring 框架。如果不存在,我的数据库将在初始化时创建。我的开发平台是intellij。

问题是当我运行应用程序时

@Autowired
private IPersonService personService; // comes null?

这是我的类和配置文件。

myDB.sql:

CREATE TABLE IF NOT EXISTS personel(
id IDENTITY AUTO_INCREMENT,
name VARCHAR(100) NOT NULL,
age VARCHAR(100));

hibernate.properties:

db.driverClassName=org.h2.Driver
db.url=jdbc:h2:~/h2SpringProject/database/SpringSample;mv_store=false;mvcc=false
db.username=admin
db.password=

这是我的 hibernate-config.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:tx="http://www.springframework.org/schema/tx"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:jdbc="http://www.springframework.org/schema/jdbc"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/jdbc http://www.springframework.org/schema/jdbc/spring-jdbc.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd">



    <context:component-scan base-package="com.springapp"/>
    <context:annotation-config/>
    <context:property-placeholder location="hibernate.properties"/>
    <tx:annotation-driven transaction-manager="transactionManager"/>

    <bean id="dataSource"
          class="org.springframework.jdbc.datasource.SingleConnectionDataSource">
        <property name="driverClassName" value="$db.driverClassName"></property>
        <property name="url" value="$db.url"></property>
        <property name="username" value="$db.username"/>
        <property name="password" value="$db.password"/>
        <property name="suppressClose" value="true"/>
    </bean>


    <jdbc:initialize-database data-source="dataSource" ignore-failures="DROPS">
        <jdbc:script location="myDb.sql"/>
    </jdbc:initialize-database>


    <bean id="hibernateCfgProperties"
          class="org.springframework.beans.factory.config.PropertiesFactoryBean">
        <property name="properties">
            <props>
                <prop key="hibernate.dialect">org.hibernate.dialect.H2Dialect</prop>
                <prop key="hibernate.hbm2ddl.auto">update</prop>
                <prop key="hibernate.show_sql">true</prop>
                <prop key="hibernate.format_sql">false</prop>
                <prop key="hibernate.use_sql_comments">true</prop>
                <prop key="hibernate.cache.use_second_level_cache">true</prop>
                <prop key="hibernate.cache.use_query_cache">true</prop>
                <prop key="hibernate.cache.region.factory_class">
                    org.hibernate.cache.ehcache.SingletonEhCacheRegionFactory
                </prop>
            </props>
        </property>
    </bean>

    <bean id="sessionFactory"
          class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
        <property name="dataSource" ref="dataSource"/>
        <property name="hibernateProperties" ref="hibernateCfgProperties"/>
        <property name="packagesToScan" value="com.springapp.model"/>
    </bean>

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

</beans>

人物类

@Entity
@Table(name = "personel")
public class Personel 

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    @Column(name = "id")
    private long id;

    @Column(name = "name")
    private String name;

    @Column(name = "age")
    private String age;
.......

IPersonDao 接口,这里是实现类

@Component
public class PersonelDaoImpl implements IPersonelDao 

    @Autowired
    private SessionFactory sessionFactory;


    public Session getCurrentSession() 
        return sessionFactory.getCurrentSession();
    

    @Override
    public void savePersonel(Personel personel) 
        getCurrentSession().saveOrUpdate(personel);
    

    @Override
    public void deletePersonel(long id) 
        getCurrentSession().delete(id);
    

    @Override
    public List<Personel> getPersonels() 
        return getCurrentSession().createQuery("from Personel").list();
    

有一个IPersonService接口,这里是实现类

@Service("PersonelService")
public class PersonelServiceImpl implements IPersonelService 

    @Autowired
    private IPersonelDao personelDao;

    @Override
    @Transactional
    public void savePersonel(Personel personel) 
        personelDao.savePersonel(personel);
    

    @Override
    @Transactional
    public void deletePersonel(long id) 
        personelDao.deletePersonel(id);
    

    @Override
    @Transactional
    public List<Personel> getPersonels() 
        return personelDao.getPersonels();
    

这是我的主要课程

public class MainApp 

    private static ApplicationContext applicationContext;

           public static void main(String[] args) 

        applicationContext = new ClassPathXmlApplicationContext("hibernate-config.xml");

    ForExample a = new ForExample();
    a.execute();
    



@Component
public class ForExample 

    @Autowired
    private IPersonelService personelService;

    public void execute()
        Personel p = new Personel();
        p.setName("thats Ok!");
        p.setAge("614345");

        personelService.savePersonel(p);
    

【问题讨论】:

class ForExample 不是 spring 托管 bean,所以不会发生 DI 我该如何解决? ForExample 类应该是spring-managed bean,从应用程序上下文中获取。在这种情况下,依赖注入将起作用。你在某处有 ApplicationContext 的实例吗?类似ApplicationContext ctx = new ClasspathApplicationContext("config.xml"); 是的,我有,在主方法 applicationContext = new ClassPathXmlApplicationContext("hibernate-config.xml"); (也是静态的) 使用 @Component 注释标记 ForExample 类,确保它落入组件扫描包中并且它应该可以工作。如果您不希望它是 Spring bean,请不要使用 @Autowired 注释并直接从上下文中获取依赖关系 ctx.getBean(IPersonService.class); 【参考方案1】:
public class MainApp 


        public static ApplicationContext applicationContext;

        private static IPersonelService personelService;


        public static void main(String[] args) 

            applicationContext = new ClassPathXmlApplicationContext("hibernate-config.xml");
            personelService = applicationContext.getBean(IPersonelService.class);

            Personel p = new Personel();
            p.setName("thatsOK!");
            p.setAge("614345");

            personelService.savePersonel(p);

        
    

因为 spring 在运行时不识别新的操作符..

【讨论】:

以上是关于我的java项目中spring事务总是出问题?的主要内容,如果未能解决你的问题,请参考以下文章

马士兵Spring-声明式事务管理-annotation

java ee项目中 spring托管的事务应该怎么设置隔离级别

java - 如何在java桌面应用程序中使用spring(事务)和hibernate创建嵌入式H2 DB?

Spring05——Spring 如何实现事务管理

为spring添加事务管理

Spring事务的传播特性和隔离级别