千家信息网

Spring的事务管理

发表于:2025-01-22 作者:千家信息网编辑
千家信息网最后更新 2025年01月22日,一、事务的基础知识数据库事务:复杂的事务要分步执行,要么整体生效、要么整体失效。必须满足:原子性、一致性、隔离性、持久性。数据并发问题:脏读:A读取了B未提交的更改数据。不可重复读:A两次读,第二次读
千家信息网最后更新 2025年01月22日Spring的事务管理

一、事务的基础知识

  1. 数据库事务:复杂的事务要分步执行,要么整体生效、要么整体失效。

  2. 必须满足:原子性、一致性、隔离性、持久性。

  3. 数据并发问题:脏读:A读取了B未提交的更改数据。

    不可重复读:A两次读,第二次读到了B已经提交的数据。(行级锁)

    幻读(虚读):A读取B新提交的新增数据。(需添加表级锁)

    第一类丢失更新:A撤销时恢复原数据把B提交的数据覆盖了。

    第二类丢失更新:A提交时覆盖了B已经提交的数据。

  4. 数据库锁机制:一般分为表锁和行锁,按并发来分有共享锁和独占锁。数据库必须在更改的行上施加独占锁;行共享锁、行独占锁、表共享锁、表共享行独占锁、表独占锁。

  5. 事务隔离级别:


6.JDBC对事务的支持


二、ThreadLocal

  1. 概念:ThreadLocal是保存线程本地化的容器,为每个使用该变量的线程分配一个独立的变量副本。

  2. 原理:通过Map来保存每个线程的变量副本,key为线程对象,值为线程的副本。

public class TestNum {      // ①通过匿名内部类覆盖ThreadLocal的initialValue()方法,指定初始值      private static ThreadLocal seqNum = new ThreadLocal() {          public Integer initialValue() {              return 0;          }      };        // ②获取下一个序列值      public int getNextNum() {          seqNum.set(seqNum.get() + 1);          return seqNum.get();      }        public static void main(String[] args) {          TestNum sn = new TestNum();          // ③ 3个线程共享sn,各自产生序列号          TestClient t1 = new TestClient(sn);          TestClient t2 = new TestClient(sn);          TestClient t3 = new TestClient(sn);          t1.start();          t2.start();          t3.start();      }        private static class TestClient extends Thread {          private TestNum sn;            public TestClient(TestNum sn) {              this.sn = sn;          }            public void run() {              for (int i = 0; i < 3; i++) {                  // ④每个线程打出3个序列值                  System.out.println("thread[" + Thread.currentThread().getName() + "] --> sn["                           + sn.getNextNum() + "]");              }          }      }  }

在spring中大部分Bean都可以声明为singleton,故而spring要对这些非线程安全bean采用Threadlocal进行封装,这样有状态的bean就能够以sigleton的方式在多线程中正常工作。spring通过Threadlocal来实现事务管理


三、spring对事务的支持

  1. spring封装了事务模版类TranscationTemplate.


  2. 事务管理主要有3个接口:PlatformTransactionManager,TransactionDefinition,TransactionStatus.

TransactionDefinition:用于描述事务的隔离级别、超时时间、等事务属性;

PlatformTransactionManager根据TransactionDefinition提供的事务属性来创建事务;就三个方法getTransaction、commit、rollback;

TransactionStatus描述激活事务的状态;


3.spring把事务管理委托给底层具体的持久化实现框架来完成,为不同的持久化框架提供了不同的PlatformTransactionManager接口的实现类。

      3)引用数据源

Hibernate

1)Hibernate配置

                    com.mysql.jdbc.Driver                jdbc:mysql://localhost/hibernate_test                root                cheng                20                1                5000                100        3000        2        true                org.hibernate.dialect.MySQL5InnoDBDialect                update                true                true                false                            

hibernate.cfg.xml文件的主要作用就是配置了一个session-factory

  1. 在session-factory中主要通过property配置一些数据库的连接信息,我们知道,spring通常会将这种数据库连接用dataSource来表示,这样一来,hibernate.cfg.xml文件中的所有跟数据库连接的都可以干掉了,直接用spring的dataSource,而dataSource也可以用c3p0、dbcp等。

  2. 在session-factory中通过property除了配置一些数据库的连接信息之外,还有一些hibernate的配置,比如方言、自动创建表机制、格式化sql等,这些信息也需要配置起来。

  3. 还有最关键的一个持久化类所在路径的配置

2)spring的sessionFactroy配置

                                                                    com.wechat.entity.po                                                        ${hibernate.hbm2ddl.auto}                ${hibernate.dialect}                ${hibernate.show_sql}                ${hibernate.format_sql}                false                        


四、声明事务

1.基于aop/tx命名空间的配置:spring在Schema的配置中添加了一个tx命名空间,在配置文件中定义事务属性。

                                                                                                                                

2.使用注解配置声明式事务

1)在xml中配置

2)在业务类上注解

@Service@Transactionalpublic class BbtForum {    public ForumDao forumDao;    public TopicDao topicDao;    public PostDao postDao;    ...}

3)@Transactional的属性

默认属性:

事务传播行为:PROPAGATION_REQUIRED

事务隔离级别:IOSLATION_DEFAULT

读写事务属性:读/写事务

超时时间:-1

回滚设置:热河运行期异常引发回滚、任何检查型异常不会引发回滚


4)spring要在具体的业务类上使用@Transactional注解


5)在方法处使用注解会覆盖类定义的注解,如果方法需要使用特殊的事务属性,可以在方法上使用注解

@Transactional(readOnly = true)public Forum getForum(int forumId) {    return forumDao.getForum(forumId);}


五、事务的一些注意点

1.使用不同的事务管理器

@Transactional("name")使用名为name的事务管理器


2.事务管理的目的是保证数据操作的事务性(原子性、一致性、隔离性、持久性)脱离了事务,DAO一样可以进行数据操作。


3.事务的传播性

PROPAGATION_REQUIERD:如果当前没有事务就创建一个,有就加进去

PROPAGATION_SUPPORTS:支持当前事务,没有就以非事务方式执行

PROPAGATION_MANDATORY:使用当前事务,没有就抛出异常

PROPAGATION_REQURES_NEW:新建事务,如果当前存在事务,就将其挂起

PROPAGATION_NOT_SUPPORTED:以非事务方式执行,如果当前有就挂起

PROPAGATION_NEVER:以非事务方式执行,如果有就挂起

PROPAGATION_NESTED:嵌套事务

相同线程中进行互相嵌套调用的事务方法工作在相同的事务中,如果在不同线程,则不同线程下事务方法工作在独立的事务中。






0