关于数据库中的事务(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

相关文章

Python调用Orator ORM进行数据库操作

《Python调用OratorORM进行数据库操作》OratorORM是一个功能丰富且灵活的PythonORM库,旨在简化数据库操作,它支持多种数据库并提供了简洁且直观的API,下面我们就... 目录Orator ORM 主要特点安装使用示例总结Orator ORM 是一个功能丰富且灵活的 python O

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景

使用Navicat工具比对两个数据库所有表结构的差异案例详解

《使用Navicat工具比对两个数据库所有表结构的差异案例详解》:本文主要介绍如何使用Navicat工具对比两个数据库test_old和test_new,并生成相应的DDLSQL语句,以便将te... 目录概要案例一、如图两个数据库test_old和test_new进行比较:二、开始比较总结概要公司存在多

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意

查询SQL Server数据库服务器IP地址的多种有效方法

《查询SQLServer数据库服务器IP地址的多种有效方法》作为数据库管理员或开发人员,了解如何查询SQLServer数据库服务器的IP地址是一项重要技能,本文将介绍几种简单而有效的方法,帮助你轻松... 目录使用T-SQL查询方法1:使用系统函数方法2:使用系统视图使用SQL Server Configu

SQL Server数据库迁移到MySQL的完整指南

《SQLServer数据库迁移到MySQL的完整指南》在企业应用开发中,数据库迁移是一个常见的需求,随着业务的发展,企业可能会从SQLServer转向MySQL,原因可能是成本、性能、跨平台兼容性等... 目录一、迁移前的准备工作1.1 确定迁移范围1.2 评估兼容性1.3 备份数据二、迁移工具的选择2.1

Python中连接不同数据库的方法总结

《Python中连接不同数据库的方法总结》在数据驱动的现代应用开发中,Python凭借其丰富的库和强大的生态系统,成为连接各种数据库的理想编程语言,下面我们就来看看如何使用Python实现连接常用的几... 目录一、连接mysql数据库二、连接PostgreSQL数据库三、连接SQLite数据库四、连接Mo

Oracle数据库如何切换登录用户(system和sys)

《Oracle数据库如何切换登录用户(system和sys)》文章介绍了如何使用SQL*Plus工具登录Oracle数据库的system用户,包括打开登录入口、输入用户名和口令、以及切换到sys用户的... 目录打开登录入口登录system用户总结打开登录入口win+R打开运行对话框,输php入:sqlp

数据库使用之union、union all、各种join的用法区别解析

《数据库使用之union、unionall、各种join的用法区别解析》:本文主要介绍SQL中的Union和UnionAll的区别,包括去重与否以及使用时的注意事项,还详细解释了Join关键字,... 目录一、Union 和Union All1、区别:2、注意点:3、具体举例二、Join关键字的区别&php

MYSQL事务死锁问题排查及解决方案

《MYSQL事务死锁问题排查及解决方案》:本文主要介绍Java服务报错日志的情况,并通过一系列排查和优化措施,最终发现并解决了服务假死的问题,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录问题现象推测 1 - 客户端无错误重试配置推测 2 - 客户端超时时间过短推测 3 - mysql 版本问