TZ_05_Spring_转账事务基于xml的开发
2024-09-06 13:15:54
事务:通过接口的动态代理加强AccountService 实现转账的事务
ApplicationContext.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd"> <!--配置代理的service -->
<bean id="proxyAccountService" factory-bean="beanFactory" factory-method="getAccountService"></bean> <!--配置beanfactory-->
<bean id="beanFactory" class="com.hdh.factory.BeanFactory">
<!-- 注入service -->
<property name="as" ref="accountService"></property>
<!-- 注入事务管理器 -->
<property name="tm" ref="txManager"></property>
</bean> <!--配置service -->
<bean id="accountService" class="com.hdh.service.impl.AccountServiceImpl">
<property name="accountDao" ref="accountDao"></property>
</bean>
<!--配置dao -->
<bean id="accountDao" class="com.hdh.dao.impl.AccountDaoImpl">
<!-- 注入QueryRunner -->
<property name="runner" ref="runner"></property>
<!-- 注入ConnectionUtils -->
<property name="connectionUtils" ref="connectionUtils"></property>
</bean> <!--配置QueryRunner -->
<bean id="runner" class="org.apache.commons.dbutils.QueryRunner"
scope="prototype"></bean> <!-- 配置Connection的工具类 ConnectionUtils -->
<bean id="connectionUtils" class="com.hdh.utils.ConnectionUtils">
<!-- 注入数据源 -->
<property name="dataSource" ref="dataSource"></property>
</bean> <!-- 配置数据源 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<!--连接数据库的必备信息 -->
<property name="driverClass" value="com.mysql.jdbc.Driver"></property>
<property name="jdbcUrl" value="jdbc:mysql://localhost:3306/spring"></property>
<property name="user" value="root"></property>
<property name="password" value="12345"></property>
</bean> <!-- 配置事务管理器-->
<bean id="txManager" class="com.hdh.utils.TransactionManager">
<!-- 注入ConnectionUtils -->
<property name="connectionUtils" ref="connectionUtils"></property>
</bean> </beans>
2.ConnectionUtils.java //获取当前线程上的连接
/**
* 连接的工具类,它用于从数据源中获取一个连接,并且实现和线程的绑定
*/
public class ConnectionUtils { private ThreadLocal<Connection> tl = new ThreadLocal<Connection>(); private DataSource dataSource; public void setDataSource(DataSource dataSource) {
this.dataSource = dataSource;
} /**
* 获取当前线程上的连接
* @return
*/
public Connection getThreadConnection() {
try{
//1.先从ThreadLocal上获取
Connection conn = tl.get();
//2.判断当前线程上是否有连接
if (conn == null) {
//3.从数据源中获取一个连接,并且存入ThreadLocal中
conn = dataSource.getConnection();
tl.set(conn);
}
//4.返回当前线程上的连接
return conn;
}catch (Exception e){
throw new RuntimeException(e);
}
} /**
* 把连接和线程解绑
*/
public void removeConnection(){
tl.remove();
}
}
3.TranscationManager.java
/**
* 和事务管理相关的工具类,它包含了,开启事务,提交事务,回滚事务和释放连接
*/
public class TransactionManager { private ConnectionUtils connectionUtils; public void setConnectionUtils(ConnectionUtils connectionUtils) {
this.connectionUtils = connectionUtils;
} /**
* 开启事务
*/
public void beginTransaction(){
try {
connectionUtils.getThreadConnection().setAutoCommit(false);
}catch (Exception e){
e.printStackTrace();
}
} /**
* 提交事务
*/
public void commit(){
try {
connectionUtils.getThreadConnection().commit();
}catch (Exception e){
e.printStackTrace();
}
} /**
* 回滚事务
*/
public void rollback(){
try {
connectionUtils.getThreadConnection().rollback();
}catch (Exception e){
e.printStackTrace();
}
} /**
* 释放连接
*/
public void release(){
try {
connectionUtils.getThreadConnection().close();//还回连接池中
connectionUtils.removeConnection();
}catch (Exception e){
e.printStackTrace();
}
}
}
4.接口的动态代理加强AccountService
/**
* 用于创建Service的代理对象的工厂
*/
public class BeanFactory { private AccountService as;
private TransactionManager tm; public void setAs(AccountService as) {
this.as = as;
} public void setTm(TransactionManager tm) {
this.tm = tm;
} public AccountService getAccountService() { return (AccountService) Proxy.newProxyInstance(as.getClass().getClassLoader(), as.getClass().getInterfaces(),
new InvocationHandler() { public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
Object invoke=null;
try {
// 开启事物
tm.beginTransaction();
invoke = method.invoke(as, args);
//事务提交
tm.commit();
} catch (Exception e) {
// 事物回滚
tm.rollback();
} finally {
// 释放连接
tm.release();
}
return invoke;
}
}); }
}
最新文章
- Java的序列化ID的作用
- phongap、APICloud、ionic等app开发平台你都知道吗?
- ScrollView
- NGUI实现ScrollView功能
- jquery 获取一组元素的选中项 - 函数、jquery获取复选框值、jquery获取单选按钮值
- iOS crash 异常捕获
- iOS-UITableView-处理cell上按钮事件(弹出警示框,页面跳转等)
- UDF2
- Canvas处理头像上传
- java_jdbc_3层 解耦
- jquery $.each遍历json数组方法
- Eclipse(MyEclipse)使用技巧——改动凝视字体大小
- codecomb 2093【牛宫】
- leetcode先刷_Path Sum
- 区别CSS中display:box;inline;none以及HTML中 <;frame>; 标签<;table>; 标签的 frame 属性
- android 5.0
- [Shell]随机数
- Markdown几个简单的例子
- C# 通过IEnumberable接口和IEnumerator接口实现自定义集合类型foreach功能
- 计时器setInterval()-慕课网
热门文章
- php断点续传之文件上传与文件下载
- Luogu P2484 [SDOI2011]打地鼠(模拟+前缀和)
- commons lang3的StringUtils中isEmpty()方法和isBlank()方法的区别
- 关于安装了sqlite对于vs的组件,重启vs后,在外面可以连接sqlite数据库,但是在建立实体模型时没有sqlite数据源的问题
- canvas石英钟
- vue中使用watch函数,当数据改变时自动引发事件
- OS -- (python)文件和目录操作方法大全
- python3-常用模块之sys
- day72test
- springboot自己实现mysql主从数据切换机制