IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 大数据 -> SQL学习十七、事务处理 -> 正文阅读

[大数据]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() 方法,默认会回滚事务。

  大数据 最新文章
实现Kafka至少消费一次
亚马逊云科技:还在苦于ETL?Zero ETL的时代
初探MapReduce
【SpringBoot框架篇】32.基于注解+redis实现
Elasticsearch:如何减少 Elasticsearch 集
Go redis操作
Redis面试题
专题五 Redis高并发场景
基于GBase8s和Calcite的多数据源查询
Redis——底层数据结构原理
上一篇文章      下一篇文章      查看所有文章
加:2022-11-05 00:34:29  更:2022-11-05 00:36:40 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年4日历 -2025/4/30 22:22:00-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码