6.5.1    使用事务

前面我们已经知道,SQLite 数据库是支持事务的,事务的特性可以保证让某一系列的操
作要么全部完成,要么一个都不会完成。那么在什么情况下才需要使用事务呢?想象以下场 景,比如你正在进行一次转账操作,银行会将转账的金额先从你的账户中扣除,然后再向收
款方的账户中添加等量的金额。看上去好像没什么问题吧?可是,如果当你账户中的金额刚
刚被扣除,这时由于一些异常原因导致对方收款失败,这一部分钱就凭空消失了!当然银行
肯定已经充分考虑到了这种情况,它会保证扣钱和收款的操作要么一起成功,要么都不会成
功,而使用的技术当然就是事务了。

接下来我们看一看如何在 Android 中使用事务吧,仍然是在 DatabaseTest 项目的基础上 进行修改。比如 Book 表中的数据都已经很老了,现在准备全部废弃掉替换成新数据,可以 先使用 delete()方法将 Book 表中的数据删除,然后再使用 insert()方法将新的数据添加到表中。
我们要保证的是,删除旧数据和添加新数据的操作必须一起完成,否则就还要继续保留原来 的旧数据。修改 activity_main.xml 中的代码,如下所示:

<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"

android:orientation="vertical" >

……

<Button
android:id="@+id/replace_data"
android:layout_width="match_parent"
android:layout_height="wrap_content" android:text="Replace
data"

/>

</LinearLayout>

可以看到,这里又添加了一个按钮,用于进行数据替换操作。然后修改 MainActivity 中 的代码,如下所示:

public class MainActivity extends Activity {

private MyDatabaseHelper dbHelper;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

dbHelper = new MyDatabaseHelper(this, "BookStore.db",
null, 2);

……

Button replaceData = (Button) findViewById(R.id.replace_data);

replaceData.setOnClickListener(new OnClickListener() {

@Override

public void onClick(View v) {

SQLiteDatabase db = dbHelper.getWritableDatabase();

db.beginTransaction(); // 开启事务

try {

db.delete("Book", null, null);

if (true) {

//
在这里手动抛出一个异常,让事务失败

throw new
NullPointerException();

}

ContentValues values = new
ContentValues(); values.put("name", "Game of Thrones");
values.put("author", "George Martin");
values.put("pages", 720); values.put("price", 20.85);
db.insert("Book", null, values);

db.setTransactionSuccessful(); // 事务已经执行成功

} catch (Exception e) {

e.printStackTrace();

} finally {

db.endTransaction(); // 结束事务

}

}

});

}

}

上述代码就是 Android 中事务的标准用法,首先调用 SQLiteDatabase 的 beginTransaction()
方法来开启一个事务,然后在一个异常捕获的代码块中去执行具体的数据库操作,当所有的
操作都完成之后,调用 setTransactionSuccessful()表示事务已经执行成功了,最后在 finally
代码块中调用 endTransaction()来结束事务。注意观察,我们在删除旧数据的操作完成后手动 抛出了一个 NullPointerException,这样添加新数据的代码就执行不到了。不过由于事务的存
在,中途出现异常会导致事务的失败,此时旧数据应该是删除不掉的。

现在可以运行一下程序并点击 Replace data 按钮,你会发现,Book 表中存在的还是之前的旧数据。然后将手动抛出异常的那行代码去除,再重新运行一下程序,此时点击一下Replace data 按钮就会将 Book 表中的数据替换成新数据了。

6.5.2    升级数据库的最佳写法

在 6.4.2 节中我们学习的升级数据库的方式是非常粗暴的,为了保证数据库中的表是最 新的,我们只是简单地在 onUpgrade()方法中删除掉了当前所有的表,然后强制重新执行了 一遍 onCreate()方法。这种方式在产品的开发阶段确实可以用,但是当产品真正上线了之后 就绝对不行了。想象以下场景,比如你编写的某个应用已经成功上线,并且还拥有了不错的 下载量。现在由于添加新功能的原因,使得数据库也需要一起升级,然后用户更新了这个版 本之后发现以前程序中存储的本地数据全部丢失了!那么很遗憾,你的用户群体可能已经流 失一大半了。

听起来好像挺恐怖的样子,难道说在产品发布出去之后还不能升级数据库了?当然不 是,其实只需要进行一些合理的控制,就可以保证在升级数据库的时候数据并不会丢失了。 下面我们就来学习一下如何实现这样的功能,你已经知道,每一个数据库版本都会对应 一个版本号,当指定的数据库版本号大于当前数据库版本号的时候,就会进入到 onUpgrade() 方法中去执行更新操作。这里需要为每一个版本号赋予它各自改变的内容,然后在onUpgrade()方法中对当前数据库的版本号进行判断,再执行相应的改变就可以了。 接着就让我们来模拟一个数据库升级的案例,还是由 MyDatabaseHelper 类来对数据库进行管理。第一版的程序要求非常简单,只需要创建一张 Book 表,MyDatabaseHelper 中的 代码如下所示:

public class MyDatabaseHelper extends SQLiteOpenHelper {

public static final String CREATE_BOOK = "create table Book ("

+ "id integer primary key autoincrement, "

+ "author text, "

+ "price real, "

+ "pages integer, "

+ "name text)";

public MyDatabaseHelper(Context context, String name, CursorFactory factory, int version) {

super(context, name, factory, version);

}

@Override

public void onCreate(SQLiteDatabase db) {

db.execSQL(CREATE_BOOK);

}

@Override

public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {

}

}

不过,几星期之后又有了新需求,这次需要向数据库中再添加一张 Category 表。于是, 修改 MyDatabaseHelper 中的代码,如下所示:

public class MyDatabaseHelper extends SQLiteOpenHelper {

public static final String CREATE_BOOK = "create table Book
("

+ "id integer primary key autoincrement, "

+ "author text, "

+ "price real, "

+ "pages integer, "

+ "name text)";

public static final String CREATE_CATEGORY = "create table Category ("

+ "id integer primary key autoincrement, "

+ "category_name text, "

+ "category_code integer)";

public
MyDatabaseHelper(Context context, String name, CursorFactory factory, int
version) {

super(context, name, factory,
version);

}

@Override

public void onCreate(SQLiteDatabase db) {

db.execSQL(CREATE_BOOK);

db.execSQL(CREATE_CATEGORY);

}

@Override

public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {

switch (oldVersion) {

case 1:

db.execSQL(CREATE_CATEGORY);

default:

}

}

}

可以看到,在 onCreate()方法里我们新增了一条建表语句,然后又在 onUpgrade()方法中 添加了一个 switch 判断,如果用户当前数据库的版本号是 1,就只会创建一张 Category 表。 这样当用户是直接安装的第二版的程序时,就会将两张表一起创建。而当用户是使用第二版 的程序覆盖安装第一版的程序时,就会进入到升级数据库的操作中,此时由于 Book 表已经 存在了,因此只需要创建一张 Category 表即可。

但是没过多久,新的需求又来了,这次要给 Book 表和 Category 表之间建立关联,需要
在 Book 表中添加一个 category_id 的字段。再次修改 MyDatabaseHelper 中的代码,如下所示:

public class MyDatabaseHelper extends SQLiteOpenHelper {

public static final String CREATE_BOOK = "create table Book
("

+ "id integer primary key autoincrement, "

+ "author text, "

+ "price real, "

+ "pages integer, "

+ "name text, "

+ "category_id integer)";

public static final String CREATE_CATEGORY = "create table Category ("

+ "id integer primary key autoincrement, "

+ "category_name text, "

+ "category_code integer)";

public
MyDatabaseHelper(Context context, String name, CursorFactory factory, int
version) {

super(context, name, factory,
version);

}

@Override

public void
onCreate(SQLiteDatabase db) { db.execSQL(CREATE_BOOK);
db.execSQL(CREATE_CATEGORY);

}

@Override

public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {

switch (oldVersion) {

case 1:

db.execSQL(CREATE_CATEGORY);

case 2:

db.execSQL("alter table Book add column category_id integer");

default:

}

}

}

可以看到,首先我们在 Book 表的建表语句中添加了一个 category_id 列,这样当用户直
接安装第三版的程序时,这个新增的列就已经自动添加成功了。然而,如果用户之前已经安 装了某一版本的程序,现在需要覆盖安装,就会进入到升级数据库的操作中。在
onUpgrade() 方法里,我们添加了一个新的 case,如果当前数据库的版本号是 2,就会执行 alter 命令来为 Book 表新增一个 category_id 列。

这里请注意一个非常重要的细节,switch 中每一个 case 的最后都是没有使用 break 的,
为什么要这么做呢?这是为了保证在跨版本升级的时候,每一次的数据库修改都能被全部执
行到。比如用户当前是从第二版程序升级到第三版程序的,那么 case 2 中的逻辑就会执行。 而如果用户是直接从第一版程序升级到第三版程序的,那么 case 1 和 case 2 中的逻辑都会执
行。使用这种方式来维护数据库的升级,不管版本怎样更新,都可以保证数据库的表结构是
最新的,而且表中的数据也完全不会丢失了。

android: SQLite 数据库的最佳实践的更多相关文章

  1. Android和PHP开发最佳实践

    Android和PHP开发最佳实践 <Android和PHP开发最佳实践>基本信息作者: 黄隽实丛书名: 移动应用开发技术丛书出版社:机械工业出版社ISBN:9787111410508上架 ...

  2. android SQLite数据库(转)

    Android数据库 之 SQLite数据库   Android数据库  一.关系型数据库SQLIte 每个应用程序都要使用数据,Android应用程序也不例外,Android使用开源的.与操作系统无 ...

  3. Google Developing for Android 三 - Performance最佳实践

    Google Developing for Android 三 - Performance最佳实践 发表于 2015-06-07   |   分类于 Android最佳实践 原文 Developing ...

  4. Google Developing for Android 二 - Memory 最佳实践 // lightSky‘Blog

    Google Developing for Android 二 - Memory 最佳实践   |   分类于 Android最佳实践 原文:Developing for Android, II Th ...

  5. Android Sqlite 数据库版本更新

      Android Sqlite 数据库版本更新 http://87426628.blog.163.com/blog/static/6069361820131069485844/ 1.自己写一个类继承 ...

  6. Android SQLite 数据库详细介绍

    Android SQLite 数据库详细介绍 我们在编写数据库应用软件时,需要考虑这样的问题:因为我们开发的软件可能会安装在很多用户的手机上,如果应用使用到了SQLite数据库,我们必须在用户初次使用 ...

  7. [Java Performance] 数据库性能最佳实践 - JPA和读写优化

    数据库性能最佳实践 当应用须要连接数据库时.那么应用的性能就可能收到数据库性能的影响. 比方当数据库的I/O能力存在限制,或者因缺失了索引而导致运行的SQL语句须要对整张表进行遍历.对于这些问题.只相 ...

  8. Android sqlite数据库存取图片信息

    Android sqlite数据库存取图片信息 存储图片:bitmap private byte[] getIconData(Bitmap bitmap){ int size = bitmap.get ...

  9. Android SQLite 数据库 增删改查操作

    Android SQLite 数据库 增删改查操作 转载▼ 一.使用嵌入式关系型SQLite数据库存储数据 在Android平台上,集成了一个嵌入式关系型数据库--SQLite,SQLite3支持NU ...

随机推荐

  1. andorid 远程存储中JDK和Volley的GET和POST方法

    在操作Volley的时候先添加volley.jar(包) <uses-permission android:name="android.permission.INTERNET" ...

  2. 设置阿里云maven中央仓库的settings.xml

    本来想找一个可用的设置文件,结果乱七八糟的,干脆自己做了一个,同时还放上了Spring的SNAPSHOT和MILESTONE/RELEASE仓库,希望能帮到一些人. <?xml version= ...

  3. ArcGIS Engine代码共享-工作空间(workspace)对象操作

    代码: public class WorkspaceHelper { public static string GISConnectionString; public static IWorkspac ...

  4. React Native填坑之旅--ListView篇

    列表显示数据,基本什么应用都是必须.今天就来从浅到深的看看React Native的ListView怎么使用.笔者写作的时候RN版本是0.34. 最简单的 //@flow import React f ...

  5. Python3学习(3)-高级篇

    Python3学习(1)-基础篇 Python3学习(2)-中级篇 Python3学习(3)-高级篇 文件读写 源文件test.txt line1 line2 line3 读取文件内容 f = ope ...

  6. python基础整理笔记(二)

    一. 列表 1. 创建实例: a = [1,2,3] b = list() 2. 主要支持的操作及其时间复杂度如下: 3. 其他 python中的列表,在内存中实际存储的形式其实是分散的存储,比较类似 ...

  7. C编程风格的人机交互 -- CSHELL (提供源码下载)

    记得上大学时,做C语言的程序都是用sdb来调试的:再后来有了gdb,同sdb差不多,不过就好用了很多.但终究还是有点遗憾.比如,程序里设计了几个函数,如果想测试下它们,就不得不再编写个测试函数,用各种 ...

  8. haskell中的let和where

    haskell中有两种定义局部变量的方法let和where,方法分别如下 roots a b c = ((-b + det) / (a2), (-b - det) / (a2)) *a*c) a2 = ...

  9. vpn与局域网冲突解决方案

    打开vpn后,所有通过网卡发出去的包都会走vpn,而不会走局域网,所以局域网无法访问,vpn为本机指定路由,让网卡把包发给vpn线路(比如10网段).如果为本机指定到达局域网的路由,访问局域网的包就知 ...

  10. Android View 如何测量

    对于Android View的测量,我们一句话总结为:"给我位置和大小,我就知道您长到那里". 为了让大家更好的理解这个结论,我这里先讲一个日常生活中的小故事:不知道大家玩过&qu ...