Dao跨事务调用实现转账功能
1.首先在数据库当中创建数据库,并且创建它的 实现类
package com.beiwo.epet.entity;
public class Account {
private int id;
private String name;
private int money;
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getMoney() {
return money;
}
public void setMoney(int money) {
this.money = money;
}
}
2.导入jar包,并且 添加到构建路径
3.在这里采用的是C3P0数据源,并且导入它的固定格式文件c3p0-config.xml
数据源
package com.beiwo.epet.util; import java.sql.Connection;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement; import javax.sql.DataSource; import com.mchange.v2.c3p0.ComboPooledDataSource; public class C3P0Util { //C3P0数据源
private static ComboPooledDataSource dataSource=new ComboPooledDataSource(); public static DataSource getDataSource() {
return dataSource;
} public static Connection getConn(){
try {
return dataSource.getConnection();
} catch (SQLException e) {
throw new RuntimeException(e);
}
} public static Connection getConn2(){
Connection conn=null; try {
conn=dataSource.getConnection();
} catch (Exception e) {
e.printStackTrace();
} return conn;
} //释放连接
public static void realease(ResultSet rs,Statement stmt,Connection conn){
try {
if(null!=rs){
rs.close();
}
} catch (Exception e) {
// TODO: handle exception
}
try {
if(null!=stmt){
stmt.close();
}
} catch (Exception e) {
// TODO: handle exception
}
try {
if(null!=conn){
conn.close();
}
} catch (Exception e) {
// TODO: handle exception
}
} }
C3P0的固定格式
<?xml version="1.0" encoding="UTF-8"?>
<c3p0-config>
<default-config>
<property name="driverClass">com.mysql.jdbc.Driver</property>
<property name="jdbcUrl">jdbc:mysql:///day05</property>
<property name="user">root</property>
<property name="password">root</property>
<property name="initialPoolSize">10</property>
<property name="maxIdleTime">30</property>
<property name="maxPoolSize">100</property>
<property name="minPoolSize">10</property>
</default-config>
</c3p0-config>
4.定义一个Dao接口(BaseDao)
package com.beiwo.epet.dao;
import com.beiwo.epet.entity.Account;
public interface AccountDao {
/**
*
* @param fromName //谁转的
* @param toName // 转给谁
* @param money //转了多少钱
* @throws Exception
*/
public void updateAccount(String fromName,String toName,int money)throws Exception;
/**
* 更新信息
* @param account
* @throws Exception
*/
public void updateAccount(Account account)throws Exception;
/**
* 获取名字
* @param name
* @return
* @throws Exception
*/
public Account findAccountByName(String name)throws Exception;
}
5.Dao的实现类
package com.beiwo.epet.dao.impl; import org.apache.commons.dbutils.QueryRunner;
import org.apache.commons.dbutils.handlers.BeanHandler; import com.beiwo.epet.dao.AccountDao;
import com.beiwo.epet.entity.Account;
import com.beiwo.epet.util.C3P0Util;
import com.beiwo.epet.util.TransactionManager; public class AccountDaoImpl implements AccountDao{ @Override
public void updateAccount(String fromName, String toName, int money) throws Exception{
QueryRunner qr=new QueryRunner(C3P0Util.getDataSource()); qr.update("UPDATE account SET money=money-? WHERE name=?",money,fromName);
qr.update("UPDATE account SET money=money+? WHERE name=?",money,toName);
} @Override
public void updateAccount(Account account) throws Exception{
QueryRunner qr=new QueryRunner();
qr.update(TransactionManager.getConnection(),"UPDATE account SET money=? WHERE name=?", account.getMoney(),account.getName());
} @Override
public Account findAccountByName(String name) throws Exception{
QueryRunner qr=new QueryRunner();
return qr.query(TransactionManager.getConnection(),"SELECT * FROM account WHERE name=?",new BeanHandler<Account>(Account.class),name); } }
6.定义一个服务层接口来实现转账功能
package com.beiwo.epet.service;
public interface AccountService {
public void transfer(String formName,String toName,int money);
}
7.实现服务层接口
package com.beiwo.epet.service.impl; import com.beiwo.epet.dao.AccountDao;
import com.beiwo.epet.dao.impl.AccountDaoImpl;
import com.beiwo.epet.entity.Account;
import com.beiwo.epet.service.AccountService;
import com.beiwo.epet.util.TransactionManager; public class AccountServiceImpl implements AccountService{ @Override
public void transfer(String formName, String toName, int money) {
AccountDao accountDao=new AccountDaoImpl(); try {
///开始一个事务,start transaction;
//获取转入和转出的账户对象
TransactionManager.startTransaction(); Account fromAccount=accountDao.findAccountByName(formName);
Account toAccount=accountDao.findAccountByName(toName); //修改账户的各自金额
fromAccount.setMoney(fromAccount.getMoney()-money);
toAccount.setMoney(toAccount.getMoney()+money); //完成转账的操作
accountDao.updateAccount(fromAccount);
//如果没有使用ThreadLocal,那么当我们添加这个异常的时候,下面的代码就不会执行,直接进入异常处理,两个事务要么都执行,要么都不执行
int i=2/0; accountDao.updateAccount(toAccount); TransactionManager.commitTransaction(); } catch (Exception e) {
try {
TransactionManager.rollbackTransaction();;//事务的回滚
} catch (Exception e2) {
e2.printStackTrace();
} }finally{
try {
TransactionManager.close();
} catch (Exception e2) {
e2.printStackTrace();
}
} } }
8.再次过程中我们需要保证Connection是同一个Connection,,所以再定义一个管理类来保证是同一个Connection
package com.beiwo.epet.util;
import java.sql.Connection;
public class TransactionManager {
private static ThreadLocal<Connection> t1=new ThreadLocal<Connection>();
//获取连接
public static Connection getConnection(){
Connection conn=t1.get();//从当前线程中取出一个连接
if(null==conn){
conn=C3P0Util.getConn();
t1.set(conn);//把conn对象放入当前线程中去
}
return conn;
}
//开始事务
public static void startTransaction(){
try {
Connection conn=getConnection();
conn.setAutoCommit(false);//从当前线程中取出连接,并开始事务
} catch (Exception e) {
e.printStackTrace();
}
}
//提交事务
public static void commitTransaction(){
try {
Connection conn=getConnection();
conn.commit();
} catch (Exception e) {
e.printStackTrace();
}
}
//回滚事务
public static void rollbackTransaction(){
try {
Connection conn=getConnection();
conn.rollback();
} catch (Exception e) {
e.printStackTrace();
}
}
//关闭连接
public static void close(){
try {
Connection conn=getConnection();
conn.close();
t1.remove();
} catch (Exception e) {
e.printStackTrace();
}
}
}
9.最后一步就是测试,在此过程中我们没有使用main函数,所以我们自定义一个测试类来测试
package com.beiwo.epet.test; import org.junit.Test; import com.beiwo.epet.service.AccountService;
import com.beiwo.epet.service.impl.AccountServiceImpl; public class TestTransfer { @Test
public void test(){
AccountService accountService=new AccountServiceImpl(); accountService.transfer("aaa", "bbb", 100); }
}
数据库中原有的数据如下:

测试之后的结果如下:

Dao跨事务调用实现转账功能的更多相关文章
- DAO跨事物调用---转账
第一步创建实体类:Entity package com.beiwo.epet.entity; public class Account { private int id; private String ...
- Atitit java c# php c++ js跨语言调用matlab实现边缘检测等功能attilax总结
Atitit java c# php c++ js跨语言调用matlab实现边缘检测等功能attilax总结 1.1. 边缘检测的基本方法Canny最常用了1 1.2. 编写matlab边缘检测代码, ...
- Jedis的八种调用方式(功能:事务,管道)
1. packagecom.irwin.redis; 2. 3. importjava.util.Arrays; 4. importjava.util.List; ...
- (二十四)JDBC应用的事务管理(转账事例)
目录 利用 Dbutils 进行事务操作(以转账为例) 转账实现方式(不优雅的方式) ThreadLocal 类 转账实现方式(优雅的方式) 利用 Dbutils 进行事务操作(以转账为例) 我们只在 ...
- AJAX跨域调用ASP.NET MVC或者WebAPI服务的解决方案
问题描述 当跨域(cross domain)调用ASP.NET MVC或者ASP.NET Web API编写的服务时,会发生无法访问的情况. 重现方式 使用模板创建一个最简单的ASP.NET Web ...
- 使用jsonp跨域调用百度js实现搜索框智能提示,并实现鼠标和键盘对弹出框里候选词的操作【附源码】
项目中常常用到搜索,特别是导航类的网站.自己做关键字搜索不太现实,直接调用百度的是最好的选择.使用jquery.ajax的jsonp方法可以异域调用到百度的js并拿到返回值,当然$.getScript ...
- 跨域调用webapi
web端跨域调用webapi 在做Web开发中,常常会遇到跨域的问题,到目前为止,已经有非常多的跨域解决方案. 通过自己的研究以及在网上看了一些大神的博客,写了一个Demo 首先新建一个webap ...
- jsp+servlet+mysql 实现简单的银行登录转账功能
jsp+servlet+mysql 实现简单的银行登录转账功能 [前期的准备] html(登录界面),servlet(处理业务逻辑),jsp(主要实现界面),mysql(实现与数据库的简单的交互)先从 ...
- 关于AJAX跨域调用ASP.NET MVC或者WebAPI服务的问题及解决方案
作者:陈希章 时间:2014-7-3 问题描述 当跨域(cross domain)调用ASP.NET MVC或者ASP.NET Web API编写的服务时,会发生无法访问的情况. 重现方式 使用模 ...
随机推荐
- Vmware虚拟机安装Ubuntu并设置root登陆
主机操作系统是win7.在Ubuntu官网下好系统镜像.iso文件,安装好Vmware workstation软件 1 安装Ubuntu系统到Vmware虚拟机: 注意下面这步是无法直接设置账号为ro ...
- .NET 版本区别,以及与 Windows 的关系
老是记不住各 Windows 版本中的 .NET 版本号,下面汇总一下: .NET Framework各版本汇总以及之间的关系 Mailbag: What version of the .NET Fr ...
- 分享api接口验证模块
一.前言 权限验证在开发中是经常遇到的,通常也是封装好的模块,如果我们是使用者,通常指需要一个标记特性或者配置一下就可以完成,但实际里面还是有许多东西值得我们去探究.有时候我们也会用一些开源的权限验证 ...
- C#开发微信门户及应用(38)--微信摇一摇红包功能
摇一摇周边红包接口是为线下商户提供的发红包功能.用户可以在商家门店等线下场所通过摇一摇周边领取商家发放的红包.我曾经在<C#开发微信门户及应用(28)--微信“摇一摇·周边”功能的使用和接口的实 ...
- JDK动态代理实现原理
之前虽然会用JDK的动态代理,但是有些问题却一直没有搞明白.比如说:InvocationHandler的invoke方法是由谁来调用的,代理对象是怎么生成的.直到看了他的文章才彻底明白,附网址:htt ...
- Java的内存分配
java内存分配 A:栈 存储局部变量 B:堆 存储所有new出来的 C:方法区(方法区的内存中) 类加载时 方法信息保存在一块称为方法区的内存中, 并不随你创建对象而随对象保存于堆中; D:本地方法 ...
- 转载:TypeScript 简介与《TypeScript 中文入门教程》
简介 TypeScript是一种由微软开发的自由和开源的编程语言.它是JavaScript的一个超集,而且本质上向这个语言添加了可选的静态类型和基于类的面向对象编程.安德斯·海尔斯伯格,C#的首席架构 ...
- infopath发布的提示“无法解析SOAP消息”(The SOAP message cannot be parsed)问题解决方案
最近发现一个列表数据过大,每次发布infopath表单提示如下错误: 后来发现一个infopath表单通过list.asmx and Formsservice.asmx来进行发布的. This err ...
- [转]ASP.NET应用程序生命周期趣谈(二)
在上回书开始的时候我们提到博客园的IIS看了一眼我的请求后就直接交给ASP.NET去处理了,并且要求ASP.NET处理完之后返回HTML以供展示. 那么我们不仅要问: 1, IIS肯定是没有眼睛 ...
- 慎用 supportedRuntime
运行环境:win7, net4.5 现象: 无法连接SQL2012数据库,提示连接超时 原因: 真正的原因: 找微软去 解决的办法: 去除多余的supportedRuntime,或者修 ...