文章目录
项目结构:
1、创建数据库表,添加记录
2、创建 service,搭建 dao,完成对象创建和注入关系
(1)service 注入 dao,在 dao 注入 JdbcTemplate,在 JdbcTemplate 注入 DataSource
3、在 dao 创建两个方法:多钱和少钱的方法,在 service 创建方法(转账的方法)
接口UserDao:
package com.rqs.spring5.dao;
public interface UserDao {
public void addMoney();
public void reduceMoney();
}
接口实现类UserDaoImpl:
package com.rqs.spring5.dao;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Repository;
@Repository
public class UserDaoImpl implements UserDao{
@Autowired
private JdbcTemplate jdbcTemplate;
//lucy转账100给mary
//少钱
@Override
public void reduceMoney() {
String sql = "update t_account set money=money-? where username=?";
jdbcTemplate.update(sql,100,"rqs");
}
//多钱
@Override
public void addMoney() {
String sql = "update t_account set money=money+? where username=?";
jdbcTemplate.update(sql,100,"lzh");
}
}
UserService:
package com.rqs.spring5.service;
import com.rqs.spring5.dao.UserDao;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
@Service
public class UserService {
//注入dao
@Autowired
private UserDao userDao;
public void accountMoney() {
userDao.reduceMoney();
userDao.addMoney();
}
}
测试:
TestAAA:
package com.rqs.spring5.test;
import com.rqs.spring5.service.UserService;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class TestBook {
@Test
public void testJdbcTemplate() {
ApplicationContext context =
new ClassPathXmlApplicationContext("bean6.xml");
UserService userService = context.getBean("userService", UserService.class);
//转账测试
userService.accountMoney();
}
}
假设双方都有6000元,在正常情况下进行测试的测试结果:
4、上面代码,如果正常执行没有问题的,但是如果代码执行过程中出现异常,有问题
例如,转账的方法出现了异常
此时假设双方都有6000元,报错的情况下进行测试,发现转账出现了问题:
(1)上面问题如何解决呢?
- 使用事务进行解决
事务操作过程
可以用Spring的事务解决上述问题,如下所示
Spring事务管理的介绍
1、事务添加到 JavaEE 三层结构里面的Service 层中(业务逻辑层)
2、在 Spring 进行事务管理操作
(1)有两种方式:编程式事务管理(通过写代码的方式,比如上图)和声明式事务管理(一般使用后者,如下第3点所示)
3、声明式事务管理
(1)基于注解方式(一般使用这种)
(2)基于 xml 配置文件方式
4、在 Spring 进行声明式事务管理,底层使用 AOP 原理
5、Spring 事务管理相关 API
(1)提供一个接口,代表事务管理器,这个接口针对不同的框架提供不同的实现类
Spring的事务管理操作
- 演示基于注解方式的声明式事务管理
- 在 spring 配置文件配置事务管理器
<!--创建事务管理器(创建事务管理实现类的对象)-->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<!--注入数据源,指向上面所创建的dataSource-->
<property name="dataSource" ref="dataSource"></property>
</bean>
- 在 spring 配置文件,开启事务注解
(1)在 spring 配置文件引入名称空间 tx
(2)开启事务注解<!--开启事务注解 指定是哪个事务管理器(即上面所创建的那个)--> <tx:annotation-driven transaction-manager="transactionManager"></tx:annotation-driven>
- 在 service 类上面(或者 service 类里面方法上面)添加事务注解
(1)@Transactional,这个注解添加到类上面,也可以添加方法上面
如果把这个注解添加类上面,这个类里面所有的方法都添加事务
如果把这个注解添加方法上面,为这个方法添加事务
经过Spring事务管理之后,假设双方都有6000元,再次增强异常进行转账测试,发现事务发生了回滚,没有出现数据异常
事务操作(声明式事务管理的参数配置)
- 在 service 类上面添加注解@Transactional,在这个注解里面可以配置事务相关参数
1.propagation:事务传播行为
什么叫事务传播行为?参考此链接
多事务方法直接进行调用,这个过程中事务是如何进行管理的
2.ioslation:事务隔离级别
(1)事务有一个特性称作隔离性,多事务操作之间不会产生影响,如果不考虑隔离性产生很多问题。
(2)有三个读的问题:脏读、不可重复读、虚(幻)读
脏读、不可重复读、幻象读概念说明:
a.脏读:指当一个事务正在访问数据,并且对数据进行了修改,而这种数据还没有提交到数据库中,这时,另外一个事务也访问这个数据,然后使用了这个数据。因为这个数据还没有提交那么另外一个事务读取到的这个数据我们称之为脏数据。依据脏数据所做的操作肯能是不正确的。
b.不可重复读:指在一个事务内,多次读同一数据。在这个事务还没有执行结束,另外一个事务也访问该同一数据,那么在第一个事务中的两次读取数据之间,由于第二个事务的修改第一个事务两次读到的数据可能是不一样的,这样就发生了在一个事物内两次连续读到的数据是不一样的,这种情况被称为是不可重复读。
c.幻读:一个事务先后读取一个范围的记录,但两次读取的纪录数不同,我们称之为幻读(两次执行同一条 select 语句会出现不同的结果,第二次读会增加一数据行,并没有说这两次执行是在同一个事务中)
解决:通过设置事务隔离级别,解决读问题
设置方法:
mysql默认的是REPEATABLE READ
3.timeout:超时时间
(1)事务需要在一定时间内进行提交,如果不提交进行回滚
(2)默认值是 -1 ,设置时间以秒单位进行计算
4.readOnly:是否只读
(1)读:查询操作,写:添加修改删除操作
(2)readOnly 默认值 false,表示可以查询,可以添加修改删除操作
(3)设置 readOnly 值是 true,设置成 true 之后,只能查询
5.rollbackFor:回滚
(1)设置出现哪些异常进行事务回滚
6.noRollbackFor:不回滚
(1)设置出现哪些异常不进行事务回滚
完全注解开发
创建配置类,使用配置类替代 xml 配置文件
package config;
import com.alibaba.druid.pool.DruidDataSource;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;
import org.springframework.transaction.annotation.EnableTransactionManagement;
import javax.sql.DataSource;
@Configuration // 配置类
@ComponentScan(basePackages = "com.rqs") // 开启组件扫描
@EnableTransactionManagement //开启事务
public class MyConfig {
//创建数据库连接池
@Bean
public DruidDataSource getDruidDataSource() {
DruidDataSource druidDataSource = new DruidDataSource();
druidDataSource.setDriverClassName("com.mysql.jdbc.Driver");
druidDataSource.setUrl("jdbc:mysql://localhost:3306/user_db");
druidDataSource.setUsername("root");
druidDataSource.setPassword("root");
return druidDataSource;
}
//创建JdbcTemplate
@Bean
//由于上面的代码,ioc容器中已经有了dataSource,可以到ioc容器中根据类型找到dataSource进行注入
public JdbcTemplate getJdbcTemplate(DataSource dataSource) {
JdbcTemplate jdbcTemplate = new JdbcTemplate();
//注入DruidDataSource对象
jdbcTemplate.setDataSource(dataSource);
return jdbcTemplate;
}
//创建事务管理器
@Bean
public DataSourceTransactionManager getDataSourceTransactionManager(DataSource dataSource) {
DataSourceTransactionManager transactionManager = new DataSourceTransactionManager();
transactionManager.setDataSource(dataSource);
return transactionManager;
}
}
完整代码
链接:https://pan.baidu.com/s/13DEhTCi2w6zJ4YQpriCfDA
提取码:7h60