SQLite数据库版本升级的管理实现

我们知道在SQLiteOpenHelper的构造方法:

super(Context context, String name, SQLiteDatabase.CursorFactory factory, int version)

中最后一个参数表示数据库的版本号.当新的版本号大于当前的version时会调用方法:

onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion)

所以我们的重点是在该方法中实现SQLite数据库版本升级的管理

当我们项目刚开始的时候第一版SQLiteOpenHelper是这样写的:

package cc.database;

import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteDatabase.CursorFactory;
import android.database.sqlite.SQLiteOpenHelper;
/**
 * Demo描述:
 * SQLite数据库版本升级的管理实现
 *
 * 参考资料:
 * http://blog.csdn.net/guolin_blog
 * Thank you very much
 */
public class DataBaseOpenHelper extends SQLiteOpenHelper {
    private final static String DATABASE_NAME="test.db";
    private static DataBaseOpenHelper mDataBaseOpenHelper;

    public static final String CREATE_PERSON=
    "create table person(personid integer primary key autoincrement,name varchar(20),phone VARCHAR(12))";

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

	//注意:
	//将DataBaseOpenHelper写成单例的.
	//否则当在一个for循环中频繁调用openHelper.getWritableDatabase()时
	//会报错,提示数据库没有执行关闭操作
	static synchronized DataBaseOpenHelper getDBInstance(Context context) {
		if (mDataBaseOpenHelper == null) {
			mDataBaseOpenHelper = new DataBaseOpenHelper(context,DATABASE_NAME,null,1);
		}
		return mDataBaseOpenHelper;
	} 

	@Override
	public void onCreate(SQLiteDatabase db) {
		db.execSQL(CREATE_PERSON);
	}

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

	}

}

在几天之后根据项目需求,需要添加一张student表,于是DataBaseOpenHelper就出现了第二版:

package cc.database;

import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteDatabase.CursorFactory;
import android.database.sqlite.SQLiteOpenHelper;

public class DataBaseOpenHelper extends SQLiteOpenHelper {
    private final static String DATABASE_NAME="test.db";
    private static DataBaseOpenHelper mDataBaseOpenHelper;

    public static final String CREATE_PERSON=
    "create table person(personid integer primary key autoincrement,name varchar(20),phone VARCHAR(12))";

    public static final String CREATE_STUDENT=
    "create table student(studentid integer primary key autoincrement,name varchar(20),phone VARCHAR(12))";

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

	//注意:
	//将DataBaseOpenHelper写成单例的.
	//否则当在一个for循环中频繁调用openHelper.getWritableDatabase()时
	//会报错,提示数据库没有执行关闭操作
	static synchronized DataBaseOpenHelper getDBInstance(Context context) {
		if (mDataBaseOpenHelper == null) {
			//改动1
			mDataBaseOpenHelper = new DataBaseOpenHelper(context,DATABASE_NAME,null,2);
		}
		return mDataBaseOpenHelper;
	} 

	@Override
	public void onCreate(SQLiteDatabase db) {
		db.execSQL(CREATE_PERSON);
		//改动2
		db.execSQL(CREATE_STUDENT);
	}

	@Override
	public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
		//改动3
		switch (oldVersion) {

		case 1:
		db.execSQL(CREATE_STUDENT);

		default:
		}
	}

}

较版本一在版本二中有三处修改的地方:

1 版本号变成了2

2 在onCreate()方法中添加了代码db.execSQL(CREATE_STUDENT);创建student表

   因为有的用户根本就没有第一版本的APP,直接从市场下载了第二版本的App。所以当然会执行onCreate()而不会执行onUpgrade()

3 在onUpgrade()做了处理:当oldVersion为1时调用db.execSQL(CREATE_STUDENT);创建student表

   因为有的用户手机上本来就有第一版本的APP,所以在App升级到第二版本时会执行onUpgrade(),不会执行onCreate()

通过这样的处理使得不同的情况下使用第二版APP时都会生成student表

又过了一个月,根据项目变更,需要给person表添加一个字段genderid,于是DataBaseOpenHelper就出现了第三版:

package cc.database;

import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteDatabase.CursorFactory;
import android.database.sqlite.SQLiteOpenHelper;

public class DataBaseOpenHelper extends SQLiteOpenHelper {
    private final static String DATABASE_NAME="test.db";
    private static DataBaseOpenHelper mDataBaseOpenHelper;

    //改动1
    public static final String CREATE_PERSON=
    "create table person(personid integer primary key autoincrement,name varchar(20),phone VARCHAR(12)),genderid integer)";

    public static final String ALTER_PERSON="alter table person add column genderid integer";

    public static final String CREATE_STUDENT=
    "create table student(studentid integer primary key autoincrement,name varchar(20),phone VARCHAR(12))";

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

	//注意:
	//将DataBaseOpenHelper写成单例的.
	//否则当在一个for循环中频繁调用openHelper.getWritableDatabase()时
	//会报错,提示数据库没有执行关闭操作
	static synchronized DataBaseOpenHelper getDBInstance(Context context) {
		if (mDataBaseOpenHelper == null) {
			//改动2
			mDataBaseOpenHelper = new DataBaseOpenHelper(context,DATABASE_NAME,null,3);
		}
		return mDataBaseOpenHelper;
	} 

	@Override
	public void onCreate(SQLiteDatabase db) {
		db.execSQL(CREATE_PERSON);
		db.execSQL(CREATE_STUDENT);
	}

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

		case 1:
		db.execSQL(CREATE_STUDENT);

		//改动3
		case 2:
	    db.execSQL(ALTER_PERSON);

		default:
		}
	}

}

较版本二在版本三中有三处修改的地方:

1 改变了CREATE_PERSON语句,在改语句中增加了一个字段genderid

   和前面的描述类似,有的用户第一次安装该APP时就直接下载了第三版

2 修改版本号为3

   应对了用户从第一版本或者第二版本升级到第三版本的情况(见下分析)

3 在onUpgrade()方法中)做了处理:当oldVersion为2时调用 db.execSQL(ALTER_PERSON);修改person表,增加genderid字段

   应对了用户从第二版本升级到第三版本的情况(见下分析)

注意一个问题:为什么这里的switch语句在每个case中没有break???

这是为了保证跨版本升级的时候每次数据库的升级都会执行到。

比如从第二版升级到第三版本,那么case 2会被执行。

比如从第一版直接升级到第三版本,那么case 1肯定会被调用,由于没有break所以会穿透switch语句又执行case 2语句继续升级,从而保证了数据的所有版本中的升级都会被执行到。

时间: 2024-11-17 14:54:26

SQLite数据库版本升级的管理实现的相关文章

Android SQLite数据库版本升级的管理实现

Android SQLite数据库版本升级的管理实现 我们知道在SQLiteOpenHelper的构造方法: super(Context context, String name, SQLiteDatabase.CursorFactory factory, int version) 中最后一个参数表示数据库的版本号.当新的版本号大于当前的version时会调用方法: onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) 所以我们

在Android中afinal框架下实现sqlite数据库版本升级的办法

     上一篇文章采用的是SQLiteOpenHelper中的onUpgrade方法实现数据库的升级.      首先获取Context: private Context mContext=this; 然后实现FinalDb内的静态方法: FinalDb.create(mContext,"afinal.db",true,2,this);    实现FinalDb的DbUpdateListener接口中的方法:       @Override public void onUpgrade

如何用Perl访问SQLite数据库

SQLite是一个零配置.无服务端.基于文件的事务型数据库系统.由于它的轻量级,自包含和紧凑的设计,所以当你想要集成数据库到你的程序中时,SQLite是一个非常不错的选择.在这篇文章中,我会展示如何用Perl脚本来创建和访问SQLite数据库.我演示的Perl代码片段是完整的,所以你可以很简单地修改并集成到你的项目中. 访问SQLite的准备 我会使用SQLite DBI Perl驱动来连接到SQLite3.因此你需要在Linux中安装它(和SQLite3一起). Debian. Ubuntu

管理-完全的本地桌面用SQLite数据库怎么样?

问题描述 完全的本地桌面用SQLite数据库怎么样? 如题,最近在做一个桌面程序,用来管理我自己的视频.图片.文档等本地资源(咱有整理癖),用SQLite合适吗? 数据库要求: 1. 不需要常驻系统 2. 不需要事先启动服务啥的... 3. 免费... 本来考虑用xml或ini的...但不好管理不说,它不安全,容易丢失或损坏... 因为数据库里也就存一些路径...最多为缩略图... 解决方案 除了sqlite,还可以用sql ce.sql server local,性能更好.不需要服务,单文件.

App版本更新时对SQLite数据库升级或者降级遇到的问题

SQLite是Android内置的一个很小的关系型数据库.SQLiteOpenHelper是一个用来辅助管理数据库创建和版本升级问题的抽象类.我们可以继承这个抽象类,实现它的一些方法来对数据库进行自定义操作.下面两个方法必须重写: public void onCreate(SQLiteDatabase db) public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) 另外SQLiteOpenHelper子类在

Android SQLite数据库增删改查操作的使用详解_Android

一.使用嵌入式关系型SQLite数据库存储数据 在Android平台上,集成了一个嵌入式关系型数据库--SQLite,SQLite3支持NULL.INTEGER.REAL(浮点数字). TEXT(字符串文本)和BLOB(二进制对象)数据类型,虽然它支持的类型只有五种,但实际上sqlite3也接受varchar(n). char(n).decimal(p,s) 等数据类型,只不过在运算或保存时会转成对应的五种数据类型. SQLite最大的特点是你可以把各种类型的数据保存到任何字段中,而不用关心字段

Android SQLite数据库彻底掌握数据存储_Android

SQLite最大的特点是你可以把各种类型的数据保存到任何字段中,而不用关心字段声明的数据类型是什么. 例如:可以在Integer类型的字段中存放字符串,或者在布尔型字段中存放浮点数,或者在字符型字段中存放日期型值.  但有一种情况例外:定义为INTEGER PRIMARY KEY的字段只能存储64位整数, 当向这种字段保存除整数以外的数据时,将会产生错误. 另外, SQLite 在解析CREATE TABLE 语句时,会忽略 CREATE TABLE 语句中跟在字段名后面的数据类型信息,如下面语

Android SQLite数据库增删改查操作的使用详解

一.使用嵌入式关系型SQLite数据库存储数据 在Android平台上,集成了一个嵌入式关系型数据库--SQLite,SQLite3支持NULL.INTEGER.REAL(浮点数字). TEXT(字符串文本)和BLOB(二进制对象)数据类型,虽然它支持的类型只有五种,但实际上sqlite3也接受varchar(n). char(n).decimal(p,s) 等数据类型,只不过在运算或保存时会转成对应的五种数据类型. SQLite最大的特点是你可以把各种类型的数据保存到任何字段中,而不用关心字段

Android开发中使用SQLite 数据库

SQPte 一个非常流行的嵌入式数据库,它支持 SQL 语言,并且只利用很少的内存就有很好的性能.此外它还是开源的,任何人都可以使用它.许多开源项目((Mozilla, PHP, Python)都使用了 SQPte. SQPte 由以下几个组件组成:SQL 编译器.内核.后端以及附件.SQPte 通过利用虚拟机和虚拟数据库引擎(VDBE),使调试.修改和扩展 SQPte 的内核变得更加方便. 图 1. SQPte 内部结构 开发中使用SQLite 数据库-android sqlite数据库"&g