最近在做微信项目,我搭建了一个基于servlet,spring3.2,hibernate4.1的框架。因为基于消息的servlet和基于业务层是分开做的,也就是先把业务层做了,再去将所有的请求转到业务层处理。所以一开始开发就用junit做测试,模拟的消息保存数据库也都能正常进行。下面列出某一个junit 的 testcase,在这个测试的例子中,我为junit配置了事务,事务也能正常提交。所以,后面的业务层写法就一直这么开发下去,也没什么问题

  

package com.cpic.inf.tools.test;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import org.springframework.test.context.transaction.TransactionConfiguration;
import org.springframework.transaction.annotation.Transactional; import com.alibaba.fastjson.JSONArray;
import com.cpic.inf.model.json.JaoFeiJson;
import com.cpic.inf.service.SendModelMsgProxy;
import com.cpic.inf.tools.exception.SendException; @RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"classpath:spring_*.xml"})
@Transactional
@TransactionConfiguration(transactionManager = "txManager", defaultRollback = false)//true:始终回滚 false:数据提交
public class TestCase { @Autowired
SendModelMsgProxy proxy; @Test
public void testModelMsg() throws SendException{
JaoFeiJson jaofei = new JaoFeiJson();
jaofei.setAccount("62260****4765");
jaofei.setAmount("200元");
jaofei.setContno("张三");
jaofei.setFinish("2014年1月30日");
jaofei.setPaycode("交通银行");
jaofei.setPeriod("第3期");
jaofei.setPolicyno("201400000009");
jaofei.setProduct("鸿发年年及其附加险");
jaofei.setRemark("如有疑问,请在”更多--小薇在线“里留言~");
jaofei.setTemplate_id("4g3_jAn3qPqhX6DR2TbU1r9-rDN2N4KazEIPLhQ3FKQ");
jaofei.setTitle("您好,您的保单续期交费成功啦~");
//jaofei.setTopcolor("#04B404");
jaofei.setTouser("ox6yJjtSe02C6b3I_Fues2WPYszk");
jaofei.setUrl("http://www.baidu.com");
String str = JSONArray.toJSON(jaofei).toString();
proxy.sendMsg(str);
}
}

后面当业务层开发完后,将工程放到web容器里跑,发送一个请求,调用到业务层方法,前面一直正常,后面发现一个问题,事务始终不提交,先看我的棕spring部分配置

<!-- 开启AOP监听 只对当前配置文件有效 -->
<aop:aspectj-autoproxy expose-proxy="true"/> <!-- 开启注解事务 只对当前配置文件有效 -->
<tx:annotation-driven transaction-manager="txManager"/> <bean id="txManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager">
<property name="sessionFactory" ref="sessionFactory"/>
</bean> <tx:advice id="txAdvice" transaction-manager="txManager">
<tx:attributes>
<tx:method name="save*" propagation="REQUIRED" />
<tx:method name="add*" propagation="REQUIRED" />
<tx:method name="create*" propagation="REQUIRED" />
<tx:method name="insert*" propagation="REQUIRED" />
<tx:method name="update*" propagation="REQUIRED" />
<tx:method name="merge*" propagation="REQUIRED" />
<tx:method name="del*" propagation="REQUIRED" />
<tx:method name="remove*" propagation="REQUIRED" />
<tx:method name="put*" propagation="REQUIRED" />
<tx:method name="use*" propagation="REQUIRED"/>
<!--hibernate4必须配置为开启事务 否则 getCurrentSession()获取不到-->
<tx:method name="get*" propagation="REQUIRED" read-only="true" />
<tx:method name="count*" 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="send*" propagation="REQUIRED" read-only="true"/>
<tx:method name="*" read-only="true" />
</tx:attributes>
</tx:advice>
<aop:config expose-proxy="true">
<!-- 只对业务逻辑层实施事务 -->
<aop:pointcut id="txPointcut" expression="execution(* com.cpic..service.*.*(..))" />
<aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut"/>
<!-- 自定义前置通知 -->
<!-- <aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut"/> -->
</aop:config> <bean id="methodAdvice" class="com.cpic.inf.service.impl.MethodBefore"></bean>

事务配置是比较清楚的,配置了aspectj的aop做为事务,我想要在所有的impl层的类里加上配置的事务,其中我现在测试的类是com.cpic.inf.service.impl.JaoFeiImpl

来看一下这个类的实现

package com.cpic.inf.service.impl;

import java.util.Map;

import org.apache.log4j.Logger;

import com.alibaba.fastjson.JSONArray;
import com.cpic.inf.model.BaseValue;
import com.cpic.inf.model.RequestModel;
import com.cpic.inf.model.in.JaoFeiModel;
import com.cpic.inf.model.mapper.JiaoFeiModelDB;
import com.cpic.inf.service.SendModelMsgABS;
import com.cpic.inf.tools.Token;
import com.cpic.inf.tools.exception.SendException; /**
* 交费成功通知
* @author wuxw
*
*/
public class JaoFeiImpl extends SendModelMsgABS { private static Logger logger = Logger.getLogger(Token.class); /** 模版顶部边框颜色*/
public static final String TOP_COLOR = "#04B404"; /** 模版字体颜色*/
public static final String COLOR = "#000000"; public String sendMsg(Map<String,String> map) throws SendException {
String token;
try {
token = Token.getToken();
} catch (Exception e1) {
throw new SendException("无法连接Token服务器");
}
RequestModel<JaoFeiModel> reqModel = null;
try{
... try{
this.save(jiaofei);
} catch(Exception e){
throw new SendException("数据库错误:" + e.getMessage());
}
} catch (Exception e) {
throw new SendException("模板参数错误!请填写正确的模板对应参数");
}
String json = JSONArray.toJSONString(reqModel);
try {
logger.info("jiaofei end");
return this.SendModelMsg(json, token);
} catch (Exception e) {
throw new SendException("网络异常,无法连接微信服务器");
}
}
}

按照我的逻辑,如果我调用这个sendMsg方法,就能够发送模板消息(注:这是一个发送微信模板消息的功能),并将这个模板消息存到数据库里。另外,用junit测试过,这个功能不存在逻辑错误,能够在junit下,发送消息成功后能正常保存到数据库里。但后面在web工程里,面前都能成功,直到this.save(jiaofei);的时候,没有看到hibernate打印出sql(show_sql=true);也就是事务没提交,为什么会这样了?

我第一反映是

<aop:pointcut id="txPointcut" expression="execution(* com.cpic..service.*.*(..))" />

这块是不是配置错了

在网上查了一下,这个表达式怎么写,发现没有写错,我如果改成其他目录的时候

this.save(jiaofei);在执行这句的时候,会报得不到session,也就是表明aop配置是正确的。

因为我的方法是sendMsg方法,事务里没有配置这个,所以我又在配置里加上了这句

<tx:method name="send*" propagation="REQUIRED" read-only="true"/>

运行的时候还是跟以前的一样,逻辑上我是没问题,但一直找不到问题出在哪里,后面在网上找了个投机的办法

在session.save();的后面加一句session.flush();

这下子,事务能正确提交了,问题暂时是解决了,因为当时数据库操作只用到了save()方法。接着赶进度。

但做为一个严谨的程序,我知道这是不对的,因为我需要的是aop来自动管理事务,而不是现在这样,操作一次就提交一次,要是遇到事务传播,这种方法就不奏效了,所以在空闲的时候,我把这个问题仔细查找一下原因。

后面我配置了一个前置消息,配置在aop中,配置是这样的

  <aop:config expose-proxy="true">
<!-- 只对业务逻辑层实施事务 -->
<aop:pointcut id="txPointcut" expression="execution(* com.cpic..service.*.*(..))" />
<!-- 自定义前置通知 -->
<aop:advisor advice-ref="methodAdvice" pointcut-ref="txPointcut"/>
</aop:config> <bean id="methodAdvice" class="com.cpic.inf.service.impl.MethodBefore"></bean>

当我执行到sendMsg()方法之前,是先进入我的前置消息的,也就是证明,不是aop的原因,事务不提交肯定是配置出了问题,我后面在这其他业务类里加个@Transactional注解,事务也提交了,所以我猜,肯定是配置出了问题,我仔细检查,发现问题出现在

<tx:method name="send*" propagation="REQUIRED" read-only="true"/>

我仔细查了查这里面每个属性的意思,因为我的理解也都只在单词字面意思上

propagation="REQUIRED":指事务传翻

read-only="true":指只读情况下,不会有事务

所以,我把read-only="true"去掉后,事务就正常了

总结:我以前搭建过很多类似的框架,也就是把几个框架整合在一起,很多配置也都是从已有的项目中照搬过来,大部分理解也都处理字面意思。所以,今天出现了这样一个问题,一时半会也解决不了,最终在不服输的态度下,还是解决了。所以,我的感受是,就像做数学题一样,还是得多练习,才能在解决问题的速度上上一个层次。希望跟我遇到同样困难的同学共勉,共同学习!

hibernate4 spring3.2 事务不提交分析的更多相关文章

  1. .NET分布式事务未提交造成6107错误或系统被挂起的问题分析定位

    问题描述: 系统中多个功能不定期出现“Unable to get error message (6107) (0).”错误,即分布式事务超时,但报出错误的部分功能根本没有使用分布式事务. 原因分析: ...

  2. 解决 spring mvc 3.0 结合 hibernate3.2 使用<tx:annotation-driven>声明式事务无法提交的问题(转载)

    1.问题复现 spring 3.0 + hibernate 3.2 spring mvc使用注解方式:service使用@service注解 事务使用@Transactional 事务配置使用 < ...

  3. Spring事务管理全面分析

    Spring 事务属性分析什么是事物  事务管理对于企业应用而言至关重要.它保证了用户的每一次操作都是可靠的,即便出现了异常的访问情况,也不至于破坏后台数据的完整性.就像银行的自助取款机,通常都能正常 ...

  4. 【原创】002 | 搭上SpringBoot事务源码分析专车

    前言 如果这是你第二次看到师长,说明你在觊觎我的美色! 点赞+关注再看,养成习惯 没别的意思,就是需要你的窥屏^_^ 专车介绍** 该趟专车是开往Spring Boot事务源码分析的专车 专车问题 为 ...

  5. 【原创】004 | 搭上SpringBoot事务诡异事件分析专车

    前言 如果这是你第二次看到师长,说明你在觊觎我的美色! 点赞+关注再看,养成习惯 没别的意思,就是需要你的窥屏^_^ 本专车系列文章 目前连载到第四篇,本专题是深入讲解Springboot源码,毕竟是 ...

  6. 对MySQL索引、锁及事务的简单分析

    一.索引的数据结构 1.二叉搜索树实现的索引 二叉搜索树如下图,它查找元素的时间复杂度为O(logn) 但如果经常出现增删操作,最后导致二叉搜索树变成线性的二叉树,这样它查找元素的时间复杂度就会变成O ...

  7. Mysql 事务隔离级别分析

    Mysql默认事务隔离级别是:REPEATABLE-READ --查询当前会话事务隔离级别mysql> select @@tx_isolation; +-----------------+ | ...

  8. [心得体会]spring事务源码分析

    spring事务源码分析 1. 事务的初始化注册(从 @EnableTransactionManagement 开始) @Import(TransactionManagementConfigurati ...

  9. Spring3之事务管理

    事务管理是企业应用开发中确保数据完整性和一致性的关键技术.对于并发和分布式坏境中从不可预期的错误中恢复来说,事务管理特别重要.Spring作为一个企业应用框架,在不同的事务管理API之上提供了一个抽象 ...

随机推荐

  1. POJ 3047 Bovine Birthday 日期定周求 泽勒公式

    标题来源:POJ 3047 Bovine Birthday 意甲冠军:.. . 思考:式 适合于1582年(中国明朝万历十年)10月15日之后的情形 公式 w = y + y/4 + c/4 - 2* ...

  2. winsock2之最简单的win socket编程

    原文:winsock2之最简单的win socket编程 server.cpp #include <WINSOCK2.H> #include <stdio.h> #pragma ...

  3. Cf 444C DZY Loves Colors(段树)

    DZY loves colors, and he enjoys painting. On a colorful day, DZY gets a colorful ribbon, which consi ...

  4. 【Java&amp;Android开源库代码分析】のandroid-async-http の开盘

          在<[Java&Android开源库代码剖析]のandroid-smart-image-view>一文中我们提到了android-async-http这个开源库,本文正 ...

  5. 让Windows 8 / 8.1 以及 Windows Server 2012 / 2012 R2的桌面,显示我的电脑图标

    cmd ->  运行[rundll32.exe shell32.dll,Control_RunDLL desk.cpl,,0],然后勾上[我的电脑]即可.

  6. AutoCAD 2012安装错误,与.net framework (1603错误)以及ms2005vc++的问题。

    首先,这是AutoCAD2012的问题.因为,如果一台计算机已经安装了这些软件,AutoCAD是无法识别出来,因此AutoCAD就只能报错.正确的做法是:如果检测到这些软件已经被安装,则需要忽略这些问 ...

  7. jquery+html三级联动下拉框及详情页面加载时的select初始化问题

    html写的三个下拉框,如下: <select name="ddlQYWZYJ" id="ddl_QYWZYJ" class="fieldsel ...

  8. OpenSUSE13.2安装MongoDB

    真是一个悲伤的故事,就是你解决过得问题没有记住,却需要再通过搜索引擎来找一遍,幸运的是曾经你做过记录,搜索帮你找到了. 这是我一个Wordpress博客整理记录的,好久没在那里更新了,两个月的时间,我 ...

  9. Spring IOC之 使用JSR 330标准注解

    从Spring 3.0开始,Spring提供了对 JSR 330标准注解的支持.这些注解可以喝Spring注解一样被扫描到.你只需要将相关的Jar包加入到你的classpath中即可. 注意:如果你使 ...

  10. [Java]利用拦截器和自定义注解做登录以及权限验证

    1.自定义注解 需要验证登录的注解 package com.etaofinance.wap.common; import java.lang.annotation.Documented; import ...