关于数据库中的事务(transaction),程序员必须记住的三点

2024-04-18 10:08

本文主要是介绍关于数据库中的事务(transaction),程序员必须记住的三点,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

            关于数据库中的事务(transaction),程序员必须记住的三点


    1. 对于程序员来说,理解事务以及并发事务的概念非常重要。下面先通过银行的转帐业务以及演示案例帮助大家建立并深入理解事务。

    以下这段代码模拟我们在银行要完成的转账业务。研读以下代码,判断哪些SQL语句必须看作一个逻辑单元,即这些语句要执行成功都得成功;若有一条语句不能成功执行,其他语句也不允许成功。为什么会有这样的要求?

    create table b_account(
        id number(19) constraint b_account_id_pk primary key,
        banlance number);
    insert into b_account values (1111,4000);
    insert into b_account values (2222,100);

    --从帐户1111转3000到帐户2222:
    update b_account set balance = balance - 3000
    where id = 1111;
    update b_account set balance = balance + 3000
    where id = 2222;

    这是因为:如果不看成一个逻辑单元,账就不平了,即:要保证收支平衡

    那么在数据库中怎样表示一些DML语句是一个逻辑单元——通过将他们定义成事务来实现。

    我们重新审视一下数据库中的数据是怎样被改变的?

    首先程序员通过一个应用程序(系统提供如sqlplus、sql developer,或同学们自己编写的)跟数据库建连接,连接成功的标志是ORACLE SERVER端多了一个进程称之为sever process(ORACLEtarena),同时系统会创建一个会话session,客户端与oracle的交互都在这个会话环境中进行,即以前所说 SQL语句的运行环境,可以通过alter session修改该环境的设置。从现在开始,更准确地说应该是事务的运行环境

    当有很多用户都要操作数据库中的数据时,ORACLE SERVER就表现为当前有很多个session,每个session当前有一个活动事务,当一个事务结束,新的事务又开始。

     对于这一类应用系统,有一个明确的定义称为OLTP(online transaction processing 联机事务处理)系统,表现为同时有很多个session,每个session上的事务都很小。即有大量的并发小事务同时发生。

     2. 并发操作案例:

案例一:

    --创建表test,插入如数据后提交:    

     create table test(
        c1 number,
        c2 number);
    insert into test values (1,1);
    insert into test values (2,2);
    insert into test values (3,3);
    commit;

session1: 

    --向test表中插入数据:   

     insert into test values (4,4);
    insert into test values (5,5);

    --查询test表:
    select * from test;

    --结果:
    C1            C2
    ---------- ----------
    1             1
    2             2
    3             3
    4             4
    5             5

session2:

   --查询test表:

    select * from test;

    --结果:
        C1           C2
        ---------- ----------
        1            1
        2            2
        3            3

    session2中看不到4,4和5,5记录的原因:事务的隔离级别是read committed,每个session可以看到的数据是已经提交了的数据和本session中正在修改的数据。session1中的操作没有提交(commmit),所以session2中看不到session1中向test表中插入的4,4和5,5。

   程序员需要记住的第一点,我们给数据库送去的是事务,而不是SQL语句,最后必须有commit或rollback语句,否则其他session看不见你修改的数据。


    如果将session1中的操作提交:

session1: 

    commit;

案例二:

然后在其他session中查询:

session1,session2,session3:

    select * from test;

    --结果:
        C1           C2
        ---------- ----------
        1            1
        2            2
        3            3
        4            4
        5            5

    在session1中对test表进行更新(update)操作:

session1:

     update test set c1 = 11
    where c2 = 1;

    --结果:    

    1 row updated.

     在session2中对test表进行更新(update)操作:

session2:

    update test set c1 = 111
    where c1 = 1;

    --结果:
    --光标不停闪烁....

    现象是提示符始终不出现,好像没有反应。

session3:

    update test set c1 = 22
    where c2 = 2;

    --结果:
    1 row updated.

    现象是session1和session3都完成了对记录的修改,session2始终在等待。session2和session3所不同的是session2和session1修改的是同一条记录,而session3修改的是另一条记录。也就是当多个session修改同一条记录时,后一个session会等待

    在并发操作时,怎样保证每一个session可以以一致的方式读取并修改数据。

表 - 1DML锁

    session2等待的原因是想在记录上加行级排他锁,但被session1占用的,只好等待,一直等到session1释放。当事务结束,即commit或rollback之后,这些DML语句加的锁就都释放了。

session1:     

    commit;

session2:

    update test set c1 = 111
    where c1 = 1;

    --结果:
    0 rows updated.

session3:    

    drop table test purge;
    --结果:
    ERROR at line 1:
    ORA-00054: resource busy and acquire with NOWAIT specified

    错误信息:资源忙(test表上有DML操作),加锁的方式是NOWAIT,所以不会等待。

session2还没commit,因此test表上还有表级共享锁。执行DDL操作时,要在表上加DDL排他锁而不能,此时并不等待,而报错放弃。

程序员需要记住的第二点,我们给数据库送去的是事务,而不是SQL语句,最后必须有commit或rollback语句,否则DML锁不释放会阻塞其他session的操作。

案例三:

    --创建表test:

    create table test(
        c1 number,
        c2 number);        

     --向表中插入数据并提交:
    insert into test values (1,1);
    insert into test values (2,2);
    insert into test values (3,3);
    commit;    

     --删除test表中c2列为3的一行:
    delete from test where c2 = 3;    

     --结果:
    1 rows deleted.

    --更新test表中c1列为1的一行:
    update test set c2 = 11 where c1 = 1;

     --结果:
    1 rows updated.

    --再次向test表中插入数据:
    insert into test values (4,4);

     --结果:
    1 rows inserted.

     --查询test表:
    select * from test;

     --结果:
    C1           C2
    ---------- ----------
    1            11
    2            2
    4            4

    --回滚:
    rollback;

    --查询test
    select * from test;

     --结果:
    C1           C2
    ---------- ----------
    1            1
    2            2
    3            3
   --原因:回滚(rollback)后,test表恢复到上一次提交时的状态,好像从未发生过提交后的操作。

    rollback是对事务的回滚,好像它从未发生,数据恢复到该事务开始之前的状态。

    数据之所以可以恢复,得益于数据库的undo segment(回滚段),会将操作之前的数据放在那里,当commit或rollback时,事务所占用的空间被释放,可以给其他事务使用。

   程序员需要记住的第三点,我们给数据库送去的是事务,而不是SQL语句,最后必须有commit或rollback语句,否则占用的回滚段空间不释放,有可能导致某些session的DML语句由于申请不到回滚空间而报错。


    用了这么多篇幅为大家阐述了关于数据库中的事务,程序员必须记住的三点,我不知道对大家有多大的帮助,但是我觉得很重要,而且薛海璐老师也重点强调了,在这儿指出来无非就是引起大家的重视!我觉得一条commit语句的重要性就像我们在编java程序时时不时要做的一个动作——Ctrl+S一样重要,有时候我们编的一个项目不断报错,花了半天时间但就是找不到错误的所在,结果到头来却是其中的一两个类没有保存!!!     

    千里之堤,溃于蚁穴!往往一个不经意的小错误就可以铸成大错!也许我们平时多留心一点,就可以省掉很多麻烦。

    细节决定成败,确实是受用终生的一句话,望大家共勉!
    


这篇关于关于数据库中的事务(transaction),程序员必须记住的三点的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

MySQL数据库双机热备的配置方法详解

《MySQL数据库双机热备的配置方法详解》在企业级应用中,数据库的高可用性和数据的安全性是至关重要的,MySQL作为最流行的开源关系型数据库管理系统之一,提供了多种方式来实现高可用性,其中双机热备(M... 目录1. 环境准备1.1 安装mysql1.2 配置MySQL1.2.1 主服务器配置1.2.2 从

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

Spring的基础事务注解@Transactional作用解读

《Spring的基础事务注解@Transactional作用解读》文章介绍了Spring框架中的事务管理,核心注解@Transactional用于声明事务,支持传播机制、隔离级别等配置,结合@Tran... 目录一、事务管理基础1.1 Spring事务的核心注解1.2 注解属性详解1.3 实现原理二、事务事

MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决

《MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决》MyBatis默认开启一级缓存,同一事务中循环调用查询方法时会重复使用缓存数据,导致获取的序列主键值均为1,... 目录问题原因解决办法如果是存储过程总结问题myBATis有如下代码获取序列作为主键IdMappe

使用Node.js和PostgreSQL构建数据库应用

《使用Node.js和PostgreSQL构建数据库应用》PostgreSQL是一个功能强大的开源关系型数据库,而Node.js是构建高效网络应用的理想平台,结合这两个技术,我们可以创建出色的数据驱动... 目录初始化项目与安装依赖建立数据库连接执行CRUD操作查询数据插入数据更新数据删除数据完整示例与最佳

Oracle数据库在windows系统上重启步骤

《Oracle数据库在windows系统上重启步骤》有时候在服务中重启了oracle之后,数据库并不能正常访问,下面:本文主要介绍Oracle数据库在windows系统上重启的相关资料,文中通过代... oracle数据库在Windows上重启的方法我这里是使用oracle自带的sqlplus工具实现的方

MySQL批量替换数据库字符集的实用方法(附详细代码)

《MySQL批量替换数据库字符集的实用方法(附详细代码)》当需要修改数据库编码和字符集时,通常需要对其下属的所有表及表中所有字段进行修改,下面:本文主要介绍MySQL批量替换数据库字符集的实用方法... 目录前言为什么要批量修改字符集?整体脚本脚本逻辑解析1. 设置目标参数2. 生成修改表默认字符集的语句3

详解Spring中REQUIRED事务的回滚机制详解

《详解Spring中REQUIRED事务的回滚机制详解》在Spring的事务管理中,REQUIRED是最常用也是默认的事务传播属性,本文就来详细的介绍一下Spring中REQUIRED事务的回滚机制,... 目录1. REQUIRED 的定义2. REQUIRED 下的回滚机制2.1 异常触发回滚2.2 回

Linux下MySQL数据库定时备份脚本与Crontab配置教学

《Linux下MySQL数据库定时备份脚本与Crontab配置教学》在生产环境中,数据库是核心资产之一,定期备份数据库可以有效防止意外数据丢失,本文将分享一份MySQL定时备份脚本,并讲解如何通过cr... 目录备份脚本详解脚本功能说明授权与可执行权限使用 Crontab 定时执行编辑 Crontab添加定

Spring 中的切面与事务结合使用完整示例

《Spring中的切面与事务结合使用完整示例》本文给大家介绍Spring中的切面与事务结合使用完整示例,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录 一、前置知识:Spring AOP 与 事务的关系 事务本质上就是一个“切面”二、核心组件三、完