浅谈oracle事务 - 竹沥半夏

标签: oracle 半夏 | 发表时间:2015-09-19 15:18 | 作者:竹沥半夏
出处:

  所谓事务,他是一个操作序列,这些操作要么都执行,要么都不执行,是一个不可分割的工作单元。通俗解释就是事务是把很多事情当成一件事情来完成,也就是大家都在一条船上,要死一起死,要活一起活。

  为什么要引入事务?事务处理可以确保除非事务性单元内的所有操作都成功完成,否则不会永久更新面向数据的资源。通过将一组相关操作组合为一个要么全部成功要么全部失败的单元,可以简化错误恢复并使应用程序更加可靠。事务结束后,能保证数据库数据的一致性。例如银行转账,一个账号扣钱,一个账号增款,要么都执行,要么都不执行,不能只执行一条,否则会出现账款错误。这样我们就可以体会到事务的重要性了吧。其实在我们的生活中,很多事情都是要组成事务来执行的。既然它那么重要,下面我们就来深入的了解一下什么是事务吧。

一.事务的四个特征ACID

  事务的四个特征非常重要,只有满足这些特征,才能保证事务执行后只有一个结果:要么成功,要么失败。

  (1)原子性(Atomicity):组成事务处理的语句形成了一个逻辑单元,不能只执行其中的一部分。

  (2)一致性(Consistemcy):事务处理前后,数据处于一致状态,保证数据的无损。

  (3)隔离性(Isolation):对数据进行修改的多个事务是彼此隔离的,互不影响。

  (4)持久性(Durability):事务完成之后,它对于系统的影响是永久的,该修改即使出现系统故障也将一直保留,真实的修改了数据库。

  示例一

---------创建表
CREATE TABLE bank
(
customerName CHAR(10), --顾客姓名
currentMoney number(10) --当前余额
);
---------添加余额不能小于1的约束
ALTER TABLE bank ADD CONSTRAINT CK_currentMoney CHECK(currentMoney >= 1);
---------插入测试数据
INSERT INTO bank(customerName, currentMoney) VALUES('张三', 1000);
INSERT INTO bank(customerName, currentMoney) VALUES('李四', 1);
commit;
select * from bank;
---------不使用事务完成转帐操作(总金额异常)
update bank set currentMoney=currentMoney-1000 where customerName='张三';
update bank set currentMoney=currentMoney+1000 where customerName='李四';
commit;
---------清空数据,
delete bank;
select * from bank;
---------使用事务完成转账操作不会出现总金额异常
set serveroutput on;
declare
li_money number(10):=0;
begin
update bank set currentMoney=currentMoney+1000 where customerName='李四';
select currentMoney into li_money from bank where customerName='李四';
dbms_output.put_line('李四当前的余额是:'||li_money);
update bank set currentMoney=currentMoney-1000 where customerName='张三';
exception
when others then
dbms_output.put_line('撤销提交');
rollback;
end;
--转帐之后查看结果,李四的钱被回滚回去了
select * from bank;

 

二.回滚到保存点

  (1)rollback:回滚到最初状态,可以参考示例一。

  (2)rollback to 保留点:回滚到保留点。

  示例二

declare
num int;
begin
savepoint a; --创建保存点a
update bank set currentmoney=currentmoney-100 where customername='张三';
savepoint b; --创建保存点b
update bank set currentmoney=currentmoney+100 where customername='李四';
savepoint c; --创建保存点c
num:=# --从提示框接收一个数据,用于操作回滚到哪一点。
insert into bank values('王五',600);
savepoint d; --创建保存点d
if num=1 then
rollback to a; --当提示框中输入1时,回滚到保存点a,即全部回滚,相当于rollback;
elsif num=2 then
rollback to b; --当提示框中输入2时,回滚到保存点b
elsif num=3 then
rollback to c; --当提示框中输入3时,回滚到保存点c
else
rollback to d; --其他情况下,回滚到保存点d
end if;
end;

 

三.事务的隔离级别

  从上面的示例中我们可以看到事务带给我们的巨大好处,保证了数据的一致性,但是,仅仅有保证用事务执行就可以保证数据前后是一致的吗?在实际情况下,事务是并发执行的,不可避免的会出现多个事务同一时间对相同数据的操作,这样就会产生各种并发问题。

  并发问题:(1)脏读:事务T1、T2,T1读取了T2已经更改但还没有提交的数据,之后,若T2回滚,T1读到的数据就是临时无效的脏数据。

        (2)不可重复读:事务T1、T2,T1读取了一个字段,然后T2更新了该字段,当T1再次读取该字段时,值就不同了。

        (3)幻读:事务T1、T2,T1读取了一个字段,T2在在该表中又插入了一些字段,当T1再次读取时,会多出几行。

  为了解决上面这些问题,引入了隔离级别,显然,隔离级别是指一个事务与其他事务的隔离程度,隔离级别都有哪些呢,下面简单介绍了一下。

隔离级别 描述
read uncommitted(读取未提交数据) 允许事务读取未被其他事务提交的变更,脏读、不可重复读和幻读问题都会出现。
read committed(读取已提交数据) 只允许事务读取已被其他事务提交的变更,可以避免脏读,但不可重复读和幻读问题仍然存在。
repeatable committed(可重复读) 确保一个事务可以多次从一个字段中读取到相同的值,避免了脏读、不可重复读,但幻读问题仍然存在。
serializble(串行化) 在这个事务执行期间,禁止其他表对该表的插入、删除、修改操作,所有并发问题都可以避免,但性能低下。

 

 

 

 

 

   Oracle 支持的 2 种事务隔离级别:READ COMMITED, SERIALIZABLE。在 Oracle 默认的事务隔离级别为: READ COMMITED。Oracle同时提供了一个Read only的隔离级别。
  虽然隔离级别可以避免并发问题,但是要酌情使用,毕竟隔离级别越高,并发性就越弱,这样软件的性能就降低了。
四.总结
  (1)事务只有一个执行结果,要么成功,要么失败。
  (2)事务的四个特点:原子性、一致性、隔离性和持续性。
  (3)事务的执行过程中发生问题时,可以直接回滚到最初未执行之前的数据状态,也可以通过savepoint设置保存点,回滚到特定的保存点位置。
  (4)当多个事务同时访问相同数据时,可能产生并发问题,为了避免这些问题,引入了隔离级别。但是隔离级别越高,虽然数据一致性越好,但是并发性却更弱。所以要酌情使用。
 

本文链接: 浅谈oracle事务,转载请注明。

相关 [oracle 半夏] 推荐:

浅谈oracle事务 - 竹沥半夏

- - 博客园_首页
  所谓事务,他是一个操作序列,这些操作要么都执行,要么都不执行,是一个不可分割的工作单元. 通俗解释就是事务是把很多事情当成一件事情来完成,也就是大家都在一条船上,要死一起死,要活一起活. 事务处理可以确保除非事务性单元内的所有操作都成功完成,否则不会永久更新面向数据的资源. 通过将一组相关操作组合为一个要么全部成功要么全部失败的单元,可以简化错误恢复并使应用程序更加可靠.

Oracle 收购 Ksplice

- feng823 - LinuxTOY
实现无需重启即可为 Linux 内核打安全补丁的 Ksplice 被 Oracle 收购. 在被收购前, Ksplice 为 Fedora, Ubuntu 免费提供该功能,对于 RHEL 和 CentOS 则需要订阅其产品. Oracle 表示将把 Ksplice 带来的零宕机安全更新功能添加到 Oracle 产品订阅服务中,同时停止对其他企业级 Linux 发行版的支持,将 Oracle Unbreakable Linux 打造成唯一具备零宕机安全更新功能的企业级 Linux 发行版.

Linux Ksplice,MySQL and Oracle

- Syn - DBA Notes
Oracle 在 7 月份收购了 Ksplice. 使用了 Ksplice 的 Linux 系统,为 Kernel 打补丁无需重启动,做系统维护的朋友应该明白这是一个杀手级特性. 现在该产品已经合并到 Oracle Linux 中. 目前已经有超过 700 家客户,超过 10 万套系统使用了 Ksplice (不知道国内是否已经有用户了.

oracle license计算

- Fenng - eagle's home
Oracle license的计算是基于CPU core的. 用core的数目乘以一个系数core factor就可以得到所需的oracle license的数目. 对于不同的CPU,core factor是不一样的,可以从oracle提供的这张列表中查到 Oracle Processor Core Factor Table.

Oracle Exadata初探

- - 技术改变世界 创新驱动中国 - 《程序员》官网
在我们看来,它是一个把硬件和软件根据合理的配置整合在一起的 Oracle数据库(在本文编写时是11gR2版本)平台. Exadata数据库机器包含了存储子系统,在存储层上运行着研发的新软件,这使得研发人员可以做一些在其他平台上无法完成的事情. 实际上,Exadata一开始是以一个存储系统形式诞生的,如果你跟参与研发此产品的人交谈,你经常会听到他们称存储组件为Exadata或者是SAGE (Storage Appliance for Grid Environments,网格环境存储设备),这是该产品研发项目的代码名称.

Oracle MySQL Or NoSQL续

- - Sky.Jian 朝阳的天空
接前面一篇,这里再将之前在“中国系统架构师大会”5周年的时候发布的纪念册“IT架构实录”上的一篇文章发出来,也算是前面博文中PPT的一个文字版解读吧. Oracle,MySQL 还是 NoSQL. 随着阿里系的“去IOE”运动在社区的宣传声越来越大,国内正在掀起一股“去xxx”的技术潮. 不仅仅是互联网企业,包括运营商以及金融机构都已经开始加入到这个潮流之中.

[转]Oracle 碎片

- - 小鸥的博客
  当生成一个数据库时,它会 分成称为表空间( Tablespace )的多个逻辑段( Segment ),如系统( System )表空间 , 临时( Temporary )表空间等. 一个表空间可以包含多个数据范围( Extent )和一个或多个自由范围块,即自由空间( Free Space ).

oracle 索引

- - 数据库 - ITeye博客
        自动:在使用primary和unique后系统会自动创建唯一索引.         手动:create   index   索引名  on 表名(字段1,....). 查询表上有哪些索引(网上找的,能用,表名和索引名要大写). 1、查找表的所有索引(包括索引名,类型,构成列):. select t.*,i.index_type from user_ind_columns t,user_indexes i where t.index_name = i.index_name and t.table_name = i.table_name and t.table_name = 要查询的表.

Oracle索引

- - Oracle - 数据库 - ITeye博客
在关系数据库中,索引是一种与表有关的数据库结构,它可以使对应于表的SQL语句执行得更快. 索引的作用相当于图书的目录,可以根据目录中的页码快速找到所需的内容. 对于数据库来说,索引是一个必选项,但对于现在的各种大型数据库来说,索引可以大大提高数据库的性能,以至于它变成了数据库不可缺少的一部分. singlecolumnorconcatenated  对一列或多列建所引.

Oracle表空间(tablespaces)

- - 博客园_首页
  我们知道oarcle数据库真正存放数据的是数据文件(data files),Oarcle表空间(tablespaces)实际上是一个逻辑的概念,他在物理上是并不存在的,那么把一组data files 捻在一起就成为一个表空间. 一个数据库可以包含多个表空间,一个表空间只能属于一个数据库. 一个表空间包含多个数据文件,一个数据文件只能属于一个表空间.