小码农的代码(二)----------SpringJDBC事务控制

2024-09-04 12:38

本文主要是介绍小码农的代码(二)----------SpringJDBC事务控制,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

今年年初遇到一个对接ERP与WMS的项目,只做后端数据同步,不进行页面展示,简而言之就是我们接收ERP http请求传送过来的数据然后转存到自己的数据库并将数据同步到WMS系统的数据库中,(ps:我也比较奇怪为什么不直接用http请求进行对接,而来中间再转存一次)。由于项目较小又不需要前端展示就直接只用SpringJDBC进行数据库操作,项目涉及的内容包括http请求接口的开发、多数据库操作,本文先拉出来整理下SpringJDBC的直接使用。 
SpringJDBC的使用目前来看非常简单,主要就是三个步骤。 
一、maven加载所需要的jar包; 
一、Spring配置文件进行数据源与jdbctemple的配置; 
二、直接在数据访问层使用。 
以下是spring配置文件的内容:
Java代码   收藏代码
  1. <!-- 引入属性文件 -->  
  2.     <context:property-placeholder location="classpath:config.properties" />  
  3.     <!-- 自动扫描dao和service包( 自动注入) -->  
  4.     <context:component-scan base-package="com.zh.demo" />  
  5.     <!-- 数据连接池C3P0 -->  
  6.     <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"  
  7.         destroy-method="close">  
  8.         <property name="driverClass" value="${jdbc.driverClass}" />  
  9.         <property name="jdbcUrl" value="${jdbc.jdbcUrl}" />  
  10.         <property name="user" value="${jdbc.user}" />  
  11.         <property name="password" value="${jdbc.password}" />  
  12.     </bean>  
  13.   
  14.     <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">  
  15.         <property name="dataSource" ref="dataSource" />  
  16.     </bean>  

这里的注意点在于spring的数据源配置,不直接使用DriverManagerDataSource的原因是由于其没有实现连接池的机制,因此使用C3P0,这时需要加载c3p0的支持JAR包; 
另外在数据访问层进行数据操作时如下
Java代码   收藏代码
  1. public class BaseDaoImpl implements BaseDao{  
  2.   
  3.     @Autowired  
  4.     private JdbcTemplate jdbcTemplate;  
  5.       
  6.     /** 
  7.      * sql = "insert into user (name,age) values(?,?)" 
  8.      * */  
  9.     public void addBySql(String sql,final Object[] params) {  
  10.         //--------------methods01------------------------------------//  
  11.         jdbcTemplate.execute(sql);  
  12.         //-----------------or methods02------------------------------//  
  13.         //just like updateBySql  
  14.     }  
  15.   
  16.     /** 
  17.      * sql = “update table_name set name=?,age=?” 
  18.      * */  
  19.     public void updateBySql(String sql,final Object[] params) {  
  20.         //--------------methods01------------------------------------//  
  21.         jdbcTemplate.update(sql,  
  22.                 new PreparedStatementSetter() {  
  23.                     public void setValues(PreparedStatement ps)  
  24.                             throws SQLException {  
  25.                         ps.setInt(1, Integer.parseInt(params[0].toString()));  
  26.                     }  
  27.                 });       
  28.         //-----------------or methods02------------------------------//  
  29.         //jdbcTemplate.update(sql, params);  
  30.     }  
  31.   
  32.     /** 
  33.      * sql = "delete user where id =? and name = ?;" 
  34.      * */  
  35.     public void deleteBySql(String sql,final Object[] params) {  
  36.         //----------just like update---------------//  
  37.         jdbcTemplate.update(sql, params);  
  38.     }  
  39.   
  40.     /** 
  41.      * RowMapper接口封装返回集合,可自定义 
  42.      * */  
  43.     public List<UserEntity> getQueryBySql(String sql) {  
  44.         return (List<UserEntity>) jdbcTemplate.query(sql,  
  45.                 new BeanPropertyRowMapper<UserEntity>(UserEntity.class));  
  46.     }  
  47.   
  48.       
  49. }  
因为Spring配置文件中已经配置,可以直接依赖注入jdbcTemplate; 
另外也可以继承JdbcDaoSupport类,来使用jdbcTemplate
,
Java代码   收藏代码
  1. public class BaseDaoImpl2 extends JdbcDaoSupport implements BaseDao{  
  2.   
  3.       
  4.     public void addBySql(String sql,final Object[] params) {  
  5.         this.getJdbcTemplate().update(sql);  
  6.     }  
  7. }  
。 
这两个类进一步论证一个接口可以有多个实现类,谁实例化则调用哪个实现类中的方法。 
具体的代码例子与测试文件可以下载附件参考,欢迎大家斧正。 
另外对于jdbcTemplate的事务控制就是使用Spring对于事务的控制,具体内容会在后续文章中补充说明。
  • SpringJDBC.rar (20.5 KB)
  • 下载次数: 3


继续上一篇文章中所述项目的代码整理学习,前面提到直接使用springJDBC的进行数据层的操作,事务控制是必须要有的,那么这里整理一下之前使用的事务控制内容。 
首先Spring的事务控制是很完善的,能够满足本人目前遇到的几乎所有项目,具体的内容就不展开论述了,因为太多了 。这里贴一张图,基本就可以理清Spring事务控制的最基本配置。本图来源于博客园
 
由于实际项目周期的问题以及个人习惯能注解就注解=。=,当然因为某种原因,这里并不是使用纯注解的方式进行事务配置,而是使用了tx拦截器的方式进行声明式事务配置(入乡随俗,充分利用Spring特色),其中运用到的aop面向切面编程后续文章中会专门对他的aspectj代理机制进行总结,这里就不对其展开代码编写试验了,当然这里的项目aspectjweaver的JAR包是肯定要添加的 。 
首先由于是直接使用SpringJdbc因而使用DataSourceTransactionManager进行事务管理
Java代码   收藏代码
  1. <bean id="transactionManager"  
  2.         class="org.springframework.jdbc.datasource.DataSourceTransactionManager">  
  3.         <property name="dataSource" ref="dataSource" />  
  4.     </bean>其次自然需要开启tx标签注解事务  
  5. [code="java<!-- 开启注解事务 只对当前配置文件有效 -->  
  6.     <tx:annotation-driven transaction-manager="transactionManager" />  
  7.   
  8.     <!-- 事务管理配置 -->  
  9.     <tx:advice id="txAdvice" transaction-manager="transactionManager">  
  10.         <tx:attributes>  
  11.             <tx:method name="save*" propagation="REQUIRED" />  
  12.             <tx:method name="create*" propagation="REQUIRED" />  
  13.             <tx:method name="insert*" propagation="REQUIRED" />  
  14.             <tx:method name="update*" propagation="REQUIRED" />  
  15.             <tx:method name="add*" propagation="REQUIRED" />  
  16.             <tx:method name="delete*" propagation="REQUIRED" />  
  17.             <tx:method name="remove*" propagation="REQUIRED" />  
  18.             <tx:method name="cancel*" propagation="REQUIRED" />  
  19.   
  20.             <tx:method name="get*" propagation="REQUIRED" read-only="true" />  
  21.             <tx:method name="query*" propagation="REQUIRED" read-only="true" />  
  22.             <tx:method name="find*" propagation="REQUIRED" read-only="true" />  
  23.             <tx:method name="load*" propagation="REQUIRED" read-only="true" />  
  24.             <tx:method name="count*" propagation="REQUIRED" read-only="true" />  
  25.             <tx:method name="list*" propagation="REQUIRED" read-only="true" />  
  26.             <tx:method name="*" read-only="true" />  
  27.         </tx:attributes>  
  28.     </tx:advice>  
最后开启Aop的监听与监听切面
Java代码   收藏代码
  1. <aop:aspectj-autoproxy expose-proxy="true" /><aop:config expose-proxy="true">  
  2.         <!-- 只对业务逻辑层实施事务 -->  
  3.         <aop:pointcut id="txPointcut"  
  4.             expression="execution(* com.zh..service.impl.*ServiceImpl.*(..))" />  
  5.         <aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut" />  
  6.     </aop:config>  
这样配置就结束了。对于不想使用切面直接使用纯注解方式后续的文章总结中可以尝试使用。 
接下去就可以展开编码测试了,在上文的基础上新建了Service类,并加入@service注解,使得调用的时候可以自动注入
Java代码   收藏代码
  1. @Service("userService")  
  2. public class UserServiceImpl implements UserService{  
  3.   
  4.     @Autowired  
  5.     private BaseDao baseDao;  
  6.       
  7.         public void saveUser(String sql) {  
  8.         try {  
  9.             baseDao.addBySql("insert into user (name,age)values('张三',13)"null);  
  10.             baseDao.addBySql("insert into user (id,name,age)values(2,'张三',13)"null);  
  11.         } catch (Exception e) {  
  12.             System.out.println("代码错误新增事务回滚");  
  13.         }  
  14.     }  
  15.   
  16. }  
如果事务正常控制则因为第二条新增语句错误(因id为2的记录已存在,可自行修改)而catch到异常并且数据库中没有新增的记录(ps.数据库很简单,就死一张user表,有id,name,age三个字段,id为主键自增)。经测试,测试结果与预测一致。 
ps:注意点:在dao层新增时,如果是以下情况则数据库直接新增前一条记录
Java代码   收藏代码
  1. public void addBySql(String sql,final Object[] params) {  
  2.                 jdbcTemplate.execute(sql);  
  3.               
  4.         }  
因而保存语句应该使用PreparedStatement,
Java代码   收藏代码
  1. jdbcTemplate.update(sql,  
  2.                 new PreparedStatementSetter() {  
  3.                     public void setValues(PreparedStatement ps)  
  4.                             throws SQLException {  
  5.                         ps.setInt(1, Integer.parseInt(params[0].toString()));  
  6.                     }  
  7.                 });   
进入jdbcTemplate.execute()方法可以发现它是直接调用Statement.execute直接执行因为事实上数据库事务已经提交而调用update方法时是使用PreparedStatement进行预编译的。 
测试代码见附件,以上内容如有错误 烦请斧正,希望能够在大家的指导下更进一步,谢谢。 
下一篇博文将会对SpringJDBC的多数据源在本项目中的实际运用进行总结。
  • SpringJDBC_事务控制_.rar (26.2 KB)
  • 下载次数: 6

这篇关于小码农的代码(二)----------SpringJDBC事务控制的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1136048

相关文章

SpringBoot请求参数接收控制指南分享

《SpringBoot请求参数接收控制指南分享》:本文主要介绍SpringBoot请求参数接收控制指南,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Spring Boot 请求参数接收控制指南1. 概述2. 有注解时参数接收方式对比3. 无注解时接收参数默认位置

利用Python调试串口的示例代码

《利用Python调试串口的示例代码》在嵌入式开发、物联网设备调试过程中,串口通信是最基础的调试手段本文将带你用Python+ttkbootstrap打造一款高颜值、多功能的串口调试助手,需要的可以了... 目录概述:为什么需要专业的串口调试工具项目架构设计1.1 技术栈选型1.2 关键类说明1.3 线程模

Python Transformers库(NLP处理库)案例代码讲解

《PythonTransformers库(NLP处理库)案例代码讲解》本文介绍transformers库的全面讲解,包含基础知识、高级用法、案例代码及学习路径,内容经过组织,适合不同阶段的学习者,对... 目录一、基础知识1. Transformers 库简介2. 安装与环境配置3. 快速上手示例二、核心模

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

Spring Security+JWT如何实现前后端分离权限控制

《SpringSecurity+JWT如何实现前后端分离权限控制》本篇将手把手教你用SpringSecurity+JWT搭建一套完整的登录认证与权限控制体系,具有很好的参考价值,希望对大家... 目录Spring Security+JWT实现前后端分离权限控制实战一、为什么要用 JWT?二、JWT 基本结构

Android实现两台手机屏幕共享和远程控制功能

《Android实现两台手机屏幕共享和远程控制功能》在远程协助、在线教学、技术支持等多种场景下,实时获得另一部移动设备的屏幕画面,并对其进行操作,具有极高的应用价值,本项目旨在实现两台Android手... 目录一、项目概述二、相关知识2.1 MediaProjection API2.2 Socket 网络

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

通过Spring层面进行事务回滚的实现

《通过Spring层面进行事务回滚的实现》本文主要介绍了通过Spring层面进行事务回滚的实现,包括声明式事务和编程式事务,具有一定的参考价值,感兴趣的可以了解一下... 目录声明式事务回滚:1. 基础注解配置2. 指定回滚异常类型3. ​不回滚特殊场景编程式事务回滚:1. ​使用 TransactionT

使用Python实现全能手机虚拟键盘的示例代码

《使用Python实现全能手机虚拟键盘的示例代码》在数字化办公时代,你是否遇到过这样的场景:会议室投影电脑突然键盘失灵、躺在沙发上想远程控制书房电脑、或者需要给长辈远程协助操作?今天我要分享的Pyth... 目录一、项目概述:不止于键盘的远程控制方案1.1 创新价值1.2 技术栈全景二、需求实现步骤一、需求