事务与转账演示

1.事务:指的是逻辑上一组操作,组成这个事务的各个执行单元,要么一起成功,要么一起失败!

2.事务的特性

  • 原子性

  • 一致性

  • 隔离性

  • 持久性

3.如果不考虑隔离性,引发安全性问题

  • 读问题:

  • 脏读: 事务A读取了事务B更新的数据,然后B回滚操作,那么A读取到的数据是脏数据

  • 不可重复读::事务 A 多次读取同一数据,事务 B 在事务A多次读取的过程中,对数据作了更新并提交,导致事务A多次读取同一数据时,结果 不一致。

  • 虚读: 系统管理员A将数据库中所有学生的成绩从具体分数改为ABCDE等级,但是系统管理员B就在这个时候插入了一条具体分数的记录,当系统管理员A改结束后发现还有一条记录没有改过来,就好像发生了幻觉一样,这就叫幻读。

  • 写问题:

  • 丢失更新:

  • 小结:不可重复读的和幻读很容易混淆,不可重复读侧重于修改,幻读侧重于新增或删除。解决不可重复读的问题只需锁住满足条件的行,解决幻读需要锁表

4.如何解决安全性问题

  • 读问题解决,设置数据库隔离级别

  • 写问题解决可以使用 悲观锁和乐观锁的方式解决

  • 通过设置事务的隔离级别

    Read uncommitted: 读未提交,不能解决任何读取的问题。

    Repeatable read: 可重复读,可解决脏读和不可重复读,但虚读、幻读可能会发生

    Read committed: 读已提交,解决脏读,虚度/幻读 和 不可重复读有可能发生。

    Serialzable : 解决所有读的问题。

Spring框架的事务管理相关的类和API

  1. PlatformTransactionManager接口 -- 平台事务管理器.(真正管理事务的类)。该接口有具体的实现类,根据不同的持久层框架,需要选择不同的实现类!

  2. TransactionDefinition接口 -- 事务定义信息.(事务的隔离级别,传播行为,超时,只读)

  3. TransactionStatus接口 -- 事务的状态

总结:上述对象之间的关系:平台事务管理器真正管理事务对象.根据事务定义的信息TransactionDefinition 进行事务管理,在管理事务中产生一些状态.将状态记录到TransactionStatus中

PlatformTransactionManager接口中实现类和常用的方法

  1. 接口的实现类

    如果使用的Spring的JDBC模板或者MyBatis框架,需要选择DataSourceTransactionManager实现类

    如果使用的是Hibernate的框架,需要选择HibernateTransactionManager实现类

  2. 该接口的常用方法

    void commit(TransactionStatus status)

    TransactionStatus getTransaction(TransactionDefinition definition)

    void rollback(TransactionStatus status)

  3. TransactionDefinition

    事务隔离级别的常量

    static int ISOLATION_DEFAULT -- 采用数据库的默认隔离级别

    static int ISOLATION_READ_UNCOMMITTED

    static int ISOLATION_READ_COMMITTED

    static int ISOLATION_REPEATABLE_READ

    static int ISOLATION_SERIALIZABLE

    2.事务的传播行为常量(不用设置,使用默认值)

    先解释什么是事务的传播行为:解决的是业务层之间的方法调用!!

    PROPAGATION_REQUIRED(默认值) -- A中有事务,使用A中的事务.如果没有,B就会开启一个新的事务,将A包含进来.(保证A,B在同一个事务中),默认值!!

    PROPAGATION_SUPPORTS -- A中有事务,使用A中的事务.如果A中没有事务.那么B也不使用事务. * PROPAGATION_MANDATORY -- A中有事务,使用A中的事务.如果A没有事务.抛出异常. * PROPAGATION_REQUIRES_NEW(记)-- A中有事务,将A中的事务挂起.B创建一个新的事务.(保证A,B没有在一个事务中) * PROPAGATION_NOT_SUPPORTED -- A中有事务,将A中的事务挂起.

    PROPAGATION_NEVER -- A中有事务,抛出异常.

    PROPAGATION_NESTED(记) -- 嵌套事务.当A执行之后,就会在这个位置设置一个保存点.如果B没有问题.执行通过.如果B出现异常,运行客户根据需求回滚(选择回滚到保存点或者是最初始状态)

    事务的使用

    银行转账功能

    pom.xml配置

    <?xml version="1.0" encoding="UTF-8"?>
    ​
    <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
      xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
      <modelVersion>4.0.0</modelVersion>
    ​
      <groupId>org.example</groupId>
      <artifactId>springT</artifactId>
      <version>1.0-SNAPSHOT</version>
      <packaging>war</packaging>
    ​
      <name>springT Maven Webapp</name>
      <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <maven.compiler.source>1.8</maven.compiler.source>
        <maven.compiler.target>1.8</maven.compiler.target>
    ​
    ​
        <spring.version>5.2.5.RELEASE</spring.version>
        <aopalliance.version>1.0</aopalliance.version>
        <aspectj.weaver>1.9.2</aspectj.weaver>
        <!-- junit版本号 -->
        <junit.version>4.5</junit.version>
      </properties>
      <dependencies>
        <!-- spring -->
        <dependency>
          <groupId>org.springframework</groupId>
          <artifactId>spring-core</artifactId>
          <version>${spring.version}</version>
        </dependency>
    ​
        <dependency>
          <groupId>org.springframework</groupId>
          <artifactId>spring-beans</artifactId>
          <version>${spring.version}</version>
        </dependency>
    ​
        <dependency>
          <groupId>org.springframework</groupId>
          <artifactId>spring-context</artifactId>
          <version>${spring.version}</version>
        </dependency>
    ​
        <dependency>
          <groupId>org.springframework</groupId>
          <artifactId>spring-expression</artifactId>
          <version>${spring.version}</version>
        </dependency>
    ​
        <dependency>
          <groupId>org.springframework</groupId>
          <artifactId>spring-tx</artifactId>
          <version>${spring.version}</version>
        </dependency>
    ​
        <dependency>
          <groupId>org.springframework</groupId>
          <artifactId>spring-web</artifactId>
          <version>${spring.version}</version>
        </dependency>
    ​
        <dependency>
          <groupId>org.springframework</groupId>
          <artifactId>spring-test</artifactId>
          <version>${spring.version}</version>
        </dependency>
    ​
        <dependency>
          <groupId>junit</groupId>
          <artifactId>junit</artifactId>
          <version>4.13</version>
          <scope>test</scope>
        </dependency>
    ​
        <!--切入点表达式的支持-->
        <dependency>
          <groupId> org.aspectj</groupId>
          <artifactId> aspectjweaver</artifactId>
          <version> 1.6.11</version>
        </dependency>
        <dependency>
          <groupId>junit</groupId>
          <artifactId>junit</artifactId>
          <version>4.12</version>
          <scope>compile</scope>
        </dependency>
    ​
        <!-- Spring-jdbc 用于配置JdbcTemplate -->
        <dependency>
          <groupId>org.springframework</groupId>
          <artifactId>spring-jdbc</artifactId>
          <version>5.2.5.RELEASE</version>
        </dependency>
    ​
        <!--c3p0-->
        <dependency>
          <groupId>com.mchange</groupId>
          <artifactId>c3p0</artifactId>
          <version>0.9.5.2</version>
        </dependency>
    ​
        <dependency>
          <groupId>mysql</groupId>
          <artifactId>mysql-connector-java</artifactId>
          <version>5.1.17</version>
        </dependency>
        <dependency>
          <groupId>org.projectlombok</groupId>
          <artifactId>lombok</artifactId>
          <version>RELEASE</version>
          <scope>compile</scope>
        </dependency>
        <dependency>
          <groupId>org.projectlombok</groupId>
          <artifactId>lombok</artifactId>
          <version>RELEASE</version>
          <scope>compile</scope>
        </dependency>
      </dependencies>
    </project>

    1.配置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"
           xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx"
           xmlns:aop="http://www.springframework.org/schema/aop"
           xsi:schemaLocation="http://www.springframework.org/schema/beans
           http://www.springframework.org/schema/beans/spring-beans.xsd
           http://www.springframework.org/schema/context
           https://www.springframework.org/schema/context/spring-context.xsd
           http://www.springframework.org/schema/tx
           http://www.springframework.org/schema/tx/spring-tx.xsd http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd">
           <!--包的扫描-->
        <context:component-scan base-package="com.hp"/>
         <!--配置读取jdbc文件-->
        <context:property-placeholder location="classpath:db.properties"/>
        <!--c3p0配置-->
        <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
                <property name="driverClass" value="${datasource.connection.driver_class}"/>
                <property name="jdbcUrl" value="${datasource.connection.url}"/>
                <property name="user" value="${datasource.connection.username}"/>
                <property name="password" value="${datasource.connection.password}"/>
        </bean>
        <!--构jdbcTemplate模板交给Spring容器管理-->
        <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
            <!--引入数据源-->
            <property name="dataSource" ref="dataSource"/>
        </bean>
        <!--事务管理采用xml配置,代码中dao层和service层采用 注解方式-->
        <!--配置事务管理器-->
        <bean id="transaction" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
              <property name="dataSource" ref="dataSource"/>
        </bean>
       <!-- 配置需要 事务的目标类(主要就是业务层的类),给目标类增强的代码配置-->
       <!-- <tx:advice id="txAdvice" transaction-manager="transaction">-->
       <!--     <tx:attributes>-->
       <!--         <tx:method name="transfer" propagation="REQUIRED"/>-->
       <!--     </tx:attributes>-->
       <!-- </tx:advice>-->
        <!--aop的配置-->
       <!-- <aop:config>-->
       <!--     <aop:pointcut id="pointcut1" expression="execution(* com.hp.service.DemoService.*(..))"/>-->
       <!--     <aop:advisor advice-ref="txAdvice" pointcut-ref="pointcut1"/>-->
       <!-- </aop:config>-->
        <!--开启注解事务-->
        <tx:annotation-driven transaction-manager="transaction"/>
    </beans>

    2.开启注解

service层

package com.hp.service;
​
import com.hp.bean.Demo;
import com.hp.dao.DemoDao;
import com.hp.util.MyException;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Service;
import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.transaction.support.TransactionCallbackWithoutResult;
import org.springframework.transaction.support.TransactionTemplate;
​
import javax.annotation.Resource;
import java.util.Map;
​
@Service
public class DemoService {
    @Resource
    DemoDao demoDao;
    @Transactional//如果是XML方式@Transactional不需要
    public void transfer(String in,Double money,String out) {
        //转出人金额
        double num = demoDao.getMoneyById(in);
        System.out.println(num);
        //转出人
        if(num != 0) {
            demoDao.update("update demo set money = money - ? where name = ?", money, in);
        }else {
            throw new MyException("余额为空");
        }
       // System.out.println(1/0);
        //收款人
        if(num >= money){
            demoDao.update("update demo set money = money + ? where name = ?",money,out);
        }else {
            throw new MyException("余额不足");
        }
    }
    public Map login(String name){
         Map demo =  demoDao.login("select * from demo where name = ?",name);
        System.out.println(demo);
          return demo;
    }
}

dao层

package com.hp.dao;
​
import com.hp.bean.Demo;
import com.hp.util.MyException;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Component;
import org.springframework.stereotype.Repository;
​
import javax.annotation.Resource;
import java.util.Map;
​
@Repository
public class DemoDao {
    @Resource
    JdbcTemplate jdbcTemplate;
    public double getMoneyById(String in){
        double d = jdbcTemplate.queryForObject("select money from demo where name=?",new Object[] {in},double.class);
        return d;
    }
​
    public int update(String sql,Object...obj){
        int i = jdbcTemplate.update(sql, obj);
        return i;
    }
    public Map login(String sql,String name){
        Map demo = jdbcTemplate.queryForMap(sql,name);
        return demo;
    }
​
}

bean实体类

package com.hp.bean;
​
import lombok.Data;
​
@Data
public class Demo {
    private Integer id;
    private String name;
    private Double money;
​
    public Demo(){
​
    }
    public Demo(Integer id, String name, Double money) {
        this.id = id;
        this.name = name;
        this.money = money;
    }
}

自定义异常类

/**
 * 自定义异常类  运行时异常
 * */
public class MyException extends  RuntimeException{
    public MyException(String msg) {
        super(msg);
    }
}

测试

import com.hp.bean.Demo;
import com.hp.service.DemoService;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
​
import javax.annotation.Resource;
import java.util.Map;
import java.util.Scanner;
​
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = "classpath:applicationContext.xml")
public class DemoTest {
    @Resource
    DemoService demoService;
    @Test
    public void test(){
        System.out.println("登录");
        demoService.login("张三");
        try {
            demoService.transfer("张三",200.00,"李四");
            System.out.println("转账成功!");
        } catch (Exception e) {
            System.out.println(e.getMessage()+",转账失败!");
        }
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值