SQL学习十七、事务处理

事务处理

使用事务处理(transaction processing),通过确保成批的 SQL 操作要么 完全执行,要么完全不执行,来维护数据库的完整性。

事务处理是一种机制, 用来管理必须成批执行的 SQL操作,保证数据库不包含不完整的操作结果。
利用事务处理,可以保证一组操作不会中途停止,它们要么完全执 行,要么完全不执行(除非明确指示)。
如果没有错误发生,整组语句提 交给(写到)数据库表;
如果发生错误,则进行回退(撤销),将数据库 恢复到某个已知且安全的状态。

  • 相关概念
    1、事务(transaction)指一组 SQL语句;
    2、回退(rollback)指撤销指定 SQL语句的过程;
    3、提交(commit)指将未存储的 SQL语句结果写入数据库表;
    4、保留点(savepoint)指事务处理中设置的临时占位符(placeholder), 可以对它发布回退(与回退整个事务处理不同)。

5、隐式提交(implicit commit)一般的 SQL语句都是针对数据库表直接执行和编写的,即提交(写或保存)操作是自动进行的。

  • 可以回退的语句
    事务处理用来管理 INSERT、UPDATE 和 DELETE 语句。

实际操作

比如,我们在新的订单表(oderlist_new)中新增订单记录,过程如下:

1、检查用户表(user)中是否有对应的用户,如果不存在就添加
2、检查供应商表(supplier_new)中是否有对应的供应商,如果不存在就添加
3、在订单表(oderlist_new)中添加一条记录和用户id、用户名、供应商id关联

如果在上述存储的过程汇总,出现某种数据库故障(如超出磁盘空间、安全限制、表锁等), 造成这个过程无法完成。那么数据库中的数据会出现什么情况?
如果在存储订单信息的时候出现的故障,就会出现不完整的订单信息,比如没有对应的用户或者供应商;
如果出现在2、3之间,就会有供应商没有供应商品(在某些业务中是合理的,某些业务中是不合理的)。

这时我们就要用到事务了。


管理事务

管理事务的关键在于将 SQL语句组分解为逻辑块,并明确规定数据何时 应该回退,何时不应该回退。

  • 1、开启事务

SQL Server 中使用BEGIN TRANSACTION
Oracle中使用SET TRANSACTION
MariaDB和 MySQL中使用START TRANSACTION

  • 2、撤销操作

ROLLBACK 命令用来回退(撤销)SQL语句,使用

DELETE FROM orderlist; 
ROLLBACK; 
  • 3、事务提交

SQL Server 中使用 COMMIT TRANSACTION
Oracle 中使用COMMIT;

  • 4、使用保留点

我们回滚的时候可以全部回滚也可以部分回滚,要支持回退部分事务,必须在事务处理块中的合适位置放置占位符。这 样,如果需要回退,可以回退到某个占位符。

在 SQL中,这些占位符称为保留点,保留点的名字可以随便取,但不能重复。

在 MariaDB、MySQL和 Oracle中 创建占位符,可使用 SAVEPOINT 语句设置保留点,如 SAVEPOINT delete1;,可以通过ROLLBACK TO delete1;回滚到对应的保留点;

在 SQL Server中,需要使用SAVE TRANSACTION 语句设置保留点,如SAVE TRANSACTION delete1;,可以通过ROLLBACK TRANSACTION delete1;回滚到对应的保留点;


在DBMS中执行事务

比如,我们现在有这样一条订单信息需要入库,按照上面从操作步骤我们可以这样写:

"菠萝"    
"14"    
"10.0"  
"20181023001"   
"30"    
"王舍"    
"2018-10-23 10:12:49.000"   
"杭州水果批发总公司" 
"文一西路275号"  
"15102725297"   
"fruithangzhou@777.com" 
"郑凯"
BEGIN TRANSACTION;
insert into user (userId,userName,`password`,loginName) values (30,'王舍','ws1234','wangshe');

insert into supplier_new (supplier,supplierAddress,supplierTel,supplierEmail,supplierContact) values ('杭州水果批发总公司','文一西路275号','15102725297','fruithangzhou@777.com','郑凯');

insert into oderlist_new
(goodsName,quantity,item_price,orderNo,userId,userName,orderTime,supplierId)
values ('菠萝',14,10.0,'20181023001',30,'王舍','2018-10-23 10:12:49.000',
(select id from supplier_new where `supplier` = '杭州水果批发总公司'));

END TRANSACTION;
执行结果

在Android中执行事务

同样是上述数据,加入到订单库中,我们按照【实际操作】中提到的过程判断在代码中实现:

private void initData() {
        File test = new File(Environment.getExternalStorageDirectory(), "DBTest");
        if (!test.exists()) {
            test.mkdirs();
        }
        String dbTest = String.format("%s/%s", test.getPath(), "task.db");
        SQLiteOpenHelper helper = new MySQLiteOpenHelper(this, dbTest, null, 3);
        SQLiteDatabase db = helper.getWritableDatabase();
        //开启事务
        db.beginTransaction();
        try {
            String[] userId = {"30"};
            String sql0 = "select * from user where userId=?";
            Cursor cursor0 = db.rawQuery(sql0, userId);
            if (!cursor0.moveToFirst()) {
                String sql1 = "insert into user (userId,userName,`password`,loginName) values (?,?,?,?)";
                db.execSQL(sql1, new String[]{"39", "王舍例", "wsl234", "wangsheli"});
            }
            String[] supplier = {"杭州有机蔬菜专供经销商"};
            String sql2 = "select * from supplier_new where supplier=?";
            Cursor cursor2 = db.rawQuery(sql2, supplier);
            if (!cursor2.moveToFirst()) {
                String sql3 = "insert into supplier_new (supplier,supplierAddress,supplierTel,supplierEmail,supplierContact) values (?,?,?,?,?)";
                db.execSQL(sql3, new String[]{"杭州有机蔬菜专供经销商", "文一西路225号", "15568432549", "organicVegetableHangZhou@999.com", "李刚"});
            }
            String sql4 = "insert into oderlist_new\n" +
                    "(goodsName,quantity,item_price,orderNo,userId,userName,orderTime,supplierId)\n" +
                    "values (?,?,?,?,?,?,?,\n" +
                    "(select id from supplier_new where supplier = ?))";
            db.execSQL(sql4, new String[]{"菠萝", "14", "10.0", "20181023001", "30", "王舍", "2018-10-23 10:12:49", "杭州有机蔬菜专供经销商"});
            db.setTransactionSuccessful();//设置事务的标志为True
        } finally {
            //结束事务,有两种情况:commit(事务的标志为True),rollback(事务的标志为False)
            db.endTransaction();
        }
    }

    /**
     * 1、实际项目中很少使用SQLiteDatabase的方法来打开数据库
     * 2、一般都是继承SQLiteOpenHelper类,来管理SQLiteDatabase
     * 3、通过SQLiteOpenHelper来获取SQLiteDatabase实例来进行相关数据库操作
     */
    private class MySQLiteOpenHelper extends SQLiteOpenHelper {

        public MySQLiteOpenHelper(Context context, String name, SQLiteDatabase.CursorFactory factory, int version) {
            super(context, name, factory, version);
        }

        @Override
        public void onCreate(SQLiteDatabase db) {
            //初次生成数据库时的回调
        }

        @Override
        public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
            //数据库版本发生改变时的回调

        }
    }
  • 用户表中没有【王舍例,userId = 39】,所以插入

  • 供应商表中有【杭州有机蔬菜专供经销商】,所以不用插入

  • 插入的订单信息

上述数据库操作,在出现异常的时候,会回滚到初始状态,即不改变数据库中的数据。我们可以在上传操作中手动加入一个异常,比如类型强转、角标越界等异常进行验证

制造一个异常

异常情况

我们可以看到在异常产生前的插入的用户数据也回滚了



笔记

1、在Android中使用事务

使用SQLiteDatabase的beginTransaction()方法可以开启一个事务,程序执行到endTransaction() 方法时会检查事务的标志是否为成功,如果为成功则提交事务,否则回滚事务。
当应用需要提交事务,必须在程序执行到endTransaction()方法之前使用setTransactionSuccessful() 方法设置事务的标志为成功,如果不调用setTransactionSuccessful() 方法,默认会回滚事务。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,590评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,808评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,151评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,779评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,773评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,656评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,022评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,678评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,038评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,756评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,411评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,005评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,973评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,053评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,495评论 2 343

推荐阅读更多精彩内容

  • ORA-00001: 违反唯一约束条件 (.) 错误说明:当在唯一索引所对应的列上键入重复值时,会触发此异常。 O...
    我想起个好名字阅读 5,169评论 0 9
  • 关于Mongodb的全面总结 MongoDB的内部构造《MongoDB The Definitive Guide》...
    中v中阅读 31,894评论 2 89
  • 蝴蝶别怪我 文/中流击水 今天起个大早,沿湖健身慢跑; 见到路边花儿,随拍几张靓照。 有的还在睡觉,有的冲着我笑;...
    楚山汉水阅读 435评论 4 9
  • 挥手,告别过去,让记忆潜藏心底,然后,道一声各自珍重。 微笑,明天你好,待风雨彩虹以后,再见,说一句,我们都好。
    梦醒时分_6c64阅读 249评论 0 0
  • 抬头一弯月,蛙鸣四五声。 灯下疾奋书,雨夜愈深沉。
    乙知己阅读 229评论 1 2