Android 数据库升级中数据保持和导入已有数据库
一.数据库升级:
在我们的程序中,或多或少都会涉及到数据库,使用数据库必定会涉及到数据库的升级,数据库升级带来的一些问题,如旧版本数据库的数据记录的保持,对新表的字段的添加等等一系列问题,还记得当我来西安的时候,面试的第二家公司,做音乐播放客户端的,就问到了这个问题;
我们开发了一个程序,当前是1.0版本。该程序用到了数据库。到1.1版本时,在数据库的某个表中增加了一个字段。那么软件1.0版本用的数据库在软件1.1版本就要被升级了。软件的1.0版本升级到1.1版本时,老的数据不能丢。那么在1.1版本的程序中就要有地方能够检测出来新的软件版本与老的数据库不兼容,并且把1.0软件的数据库升级到1.1软件能够使用的数据库。也就是说,要在1.0软件的数据库的那个表中增加那个字段,并赋予这个字段默认值。
程序如何知道我们的数据库需要升级呢?SQLiteOpenHelper类的构造函数有一个参数是version即数据库版本号。比如在软件1.0版本中,我们使用SQLiteOpenHelper访问数据库时,该参数为1,那么数据库版本号1就会写在我们的数据库中。到了1.1版本,我们的数据库需要发生变化,那么我们1.1版本的程序中就要使用一个大于1的整数来构造SQLiteOpenHelper类,用于访问新的数据库,比如2。当我们的1.1新程序读取1.0版本的老数据库时,就发现老数据库里存储的数据库版本是1,而我们新程序访问它时填的版本号为2,系统就知道数据库需要升级。
当系统在构造SQLiteOpenHelper类的对象时,如果发现版本号不一样,就会自动调用onUpgrade函数,在这个方法里对数据库进行升级。在这个函数中把老版本数据库的相应表中增加字段,并给每条记录增加默认值即可。新版本号和老版本号都会作为onUpgrade函数的参数传进来,便于开发者知道数据库应该从哪个版本升级到哪个版本。升级完成后,数据库会自动存储最新的版本号为当前数据库版本号。
SQLite提供了ALTER TABLE命令,允许用户重命名或添加新的字段到已有表中,但是不能从表中删除字段。并且只能在表的末尾添加字段,比如,为Test添加一个字段:"ALTER TABLE Test ADDCOLUMN age"
下面是我的一个测试:
public class DataSQL extends SQLiteOpenHelper {
public DataSQL(Context context, int version) {
super(context, "Test", null, version);
}
@Override
public void onCreate(SQLiteDatabase db) {
db.execSQL("create table test(id integer primary key autoincrement, name)");
}
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
if (newVersion == 2) {
db.execSQL("ALTER TABLE test ADD COLUMN age");
Cursor cr = db.rawQuery("select * from test", null);
while (cr.moveToNext()) {
String name = cr.getString(cr.getColumnIndex("name"));
ContentValues values = new ContentValues();
values.put("name", name);
values.put("age", 23);
db.update("test", values, "name=?", new String[] { name });
}
}
}
}
添加数据及读取数据
DataSQL sql = new DataSQL(this, 2);
SQLiteDatabase db = sql.getWritableDatabase();
Cursor cr = db.query("test", null, null, null, null, null, null);
while (cr.moveToNext()) {
Log.e("cr-name", "" + cr.getString(cr.getColumnIndex("name")));
Log.e("cr-age", "" + cr.getInt(cr.getColumnIndex("age")));
}
我只添加了一条数据,可以看到这条数据被打印出来了.
如果遇到复杂的修改操作,比如在修改的同时,需要进行数据的转移,那么可以采取在一个事务中执行如下语句来实现修改表的需求。
1. 将表名改为临时表
ALTERTABLE Test RENAME TO _Test;
2. 创建新表
CREATETABLE Test(id VARCHAR(32) PRIMARYKEY ,UserName VARCHAR(32) NOTNULL , Age VARCHAR(16) NOTNULL);
3. 导入数据
INSERTINTO Test SELECT id, “”, Age FROM _Test;
或者
INSERTINTO Test() SELECT id, “”, Age FROM _Test;
* 注意 双引号”” 是用来补充原来不存在的数据的!
4. 删除临时表
DROPTABLE _Test;
通过以上四个步骤,就可以完成旧数据库结构向新数据库结构的迁移,并且其中还可以保证数据不会因为升级而流失。
当然,如果遇到减少字段的情况,也可以通过创建临时表的方式来实现。
下面仍然通过一个例子来进行测试:
1.修改DataSQL的onUpgrade方法
if (newVersion == 3) {
char str = '"';
db.beginTransaction();
db.execSQL("ALTER TABLE test RENAME TO _Test");
db.execSQL("CREATE TABLE test(id integer primary key autoincrement , PassWord VARCHAR(20) NOT NULL,"
+ " UserName VARCHAR(32) NOT NULL , Age VARCHAR(16) NOT NULL)");
db.execSQL("INSERT INTO test SELECT id, " + str + str
+ ", name, age FROM _Test");
db.setTransactionSuccessful();
db.endTransaction();
}
2.修改Activity中打印信息
Log.e("cr-name", "" + cr.getString(cr.getColumnIndex("UserName")));
Log.e("cr-age", "" + cr.getInt(cr.getColumnIndex("Age")));
Log.e("cr-password", "" + cr.getInt(cr.getColumnIndex("PassWord")));

在实际开发工作中,我们的处理可能比上面所述的复杂;
假如我们开发的程序已经发布了两个版本:V1.0,V1.2,我们正在开发V1.3。每一版的数据库版本号分别是8,9,10。对于这种情况,我们应该如何实现升级?
用户的选择有:
1) V1.0 -> V1.3 DB 8 -> 10
2) V1.2 -> V1.3 DB 9 -> 10
3)注意:数据库的每一个版本所代表的数据库必须是定义好的,比如说V1.0的数据库,它可能只有两张表TableA和TableB,如果V1.2要添加一张表TableC,如果V1.3要修改TableC,那么每一个版本所对应的数据库结构如下:
V1.0 ---> TableA, TableB
V1.2 ---> TableA, TableB, TableC
V1.3 ---> TableA, TableB, TableC (Modify)
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion)
{
int upgradeVersion = oldVersion;
if (8 == upgradeVersion) {
// Create table C
String sql = "CREATE TABLE ...";
db.execSQL(sql);
upgradeVersion = 9;
}
if (9 == upgradeVersion) {
// Modify table C
upgradeVersion = 10;
}
if (upgradeVersion != newVersion) {
// Drop tables
db.execSQL("DROP TABLE IF EXISTS " + tableName);
// Create tables
onCreate(db);
}
}
在onUpgrade()方法中,处理了数据库版本从8 -> 10的升级过程,这样做的话,不论用户从8 -> 10,还是从9 - 10,最终程序的数据库都能升级到V1.3所对应的数据库结构。
二.导入已有数据库
在有的情况下,我们要在程序一开始运行的时候就导入某些固定的数据,而这些数据过大,又不可能直接代码写死,此时就需要通过导入已有数据库的方法导入数据,我们知道raw文件夹下的东西,android会原封不动的拷贝到程序中,而不会转换为二进制文件,所以,我们把数据库放到raw文件夹下供程序导入使用;
public class DBImporter {
public static final String PACKAGE_NAME = "com.example.sql";
public static final String DB_NAME = "xxx.db";
public static String DB_PATH = "/data/data/" + PACKAGE_NAME;
private Context context;
public DBImporter(Context mContext) {
this.context = mContext;
}
public SQLiteDatabase openDataBase() {
return SQLiteDatabase.openOrCreateDatabase(DB_PATH + "/" + DB_NAME, null);
}
public void copyDB() {
File file = new File(DB_PATH + "/" + DB_NAME);
if (!file.exists()) {
try {
FileOutputStream out = new FileOutputStream(file);
int buffer = 400000;
// 读取数据库并保存到data/data/packagename/xx.db...
InputStream ins = context.getResources().openRawResource(R.raw.sql_);
byte[] bts = new byte[buffer];
int length;
while ((length = ins.read(bts)) > 0) {
out.write(bts, 0, bts.length);
}
out.close();
ins.close();
SQLiteDatabase.openOrCreateDatabase(DB_PATH + "/" + DB_NAME, null);
} catch (FileNotFoundException e) {
} catch (IOException e) {
}
}
}
}
接下来便是在需要使用到该数据库的地方调用
DBImporter importer = new DBImporter(this);
importer.copyDB();
SQLiteDatabase db = importer.openDataBase(); // 获取到数据库对象,接下来便可操作了~
Android 数据库升级中数据保持和导入已有数据库的更多相关文章
- Android中数据存储(三)——SQLite数据库存储数据
当一个应用程序在Android中安装后,我们在使用应用的过程中会产生很多的数据,应用都有自己的数据,那么我们应该如何存储数据呢? 数据存储方式 Android 的数据存储有5种方式: 1. Share ...
- Android数据库升级,数据不丢失解决方案
假设要更新TableC表,建议的做法是: 1) 将TableC重命名为TableC_temp SQL语句可以这样写:ALERT TABLE TableC RENAME TO TableC_temp; ...
- Android开发8:数据存储(二)——SQLite数据库和ContentProvider的使用
前言 啦啦啦各位小伙伴们许久不见了~学期末和过年期间自己忙着做其他事没能及时更新Android开发系列课程的博客,实在是罪过罪过~ 好啦~废话不多说,进入我们今天的主题.今天我们将和大家学习其他的数据 ...
- mysql 导入 csv文件中数据,只能导入第一行
用workbench导入csv数据,只能导入数据的第一行,也就是标注每一列的列名的那一行.但问题是,每次导入完成时,系统提示已经导入了500条记录(这个文件中的确有500条记录),可是刷新数据库后打开 ...
- 我们在删除SQL Sever某个数据库表中数据的时候,希望ID重新从1开始,而不是紧跟着最后一个ID开始需要的命令
一.如果数据重要,请先备份数据 二.删除表中数据 SQL: Delete From ('表名') 如:Delete From abcd 三.执行新语句 SQL: dbcc checkident('表 ...
- Android开发——fragment中数据传递与刷新UI(更改控件)
数据传递: 1.通过数据库进行数据的传递 如在fragment中将数据保存在数据库中,之后其他的fragment或者activity直接读取数据库中的数据,数据库使用还算简单,这里就不多说,建议使用l ...
- MySQL----DQL(查询数据库表中数据)
##DQL:查询表中的记录 1.语法: select 字段列名 from 表名列表 where 条件列表 group by 分组字段 having 分组之后的条件 order by 排序 lim ...
- Android数据库升级实例
第一部分 Andoird的SQLiteOpenHelper类中有一个onUpgrade方法.帮助文档中只是说当数据库升级时该方法被触发.经过实践,解决了我一连串的疑问: 1. 帮助文档里说的“数据库升 ...
- Android数据库升级、降级、创建(onCreate() onUpgrade() onDowngrade())[4]
数据库版本升级对软件的管理操作. 我们手机经常会收到xxx软件升级什么的提醒,你的软件版本更新,同时你的数据库对应的版本也要相应的更新. 数据库版本更新需要主要的问题: 软件的1.0版本升级到1.1版 ...
随机推荐
- Redis系列二 Redis数据库介绍
1.SELECT命令 通过查看配置文件可以知道Redis默认有17个库,从0-16. 默认是在0号库.选择库使用SELECT <dbid>命令.例如选择0号库 SELECT 0 2.DB ...
- python 3.6 setup
1.添加python3.6安装包,并且安装 sudo apt-get install software-properties-common 2.下载python3.6 sudo add-apt-rep ...
- C#中Mutex的用法
C#中Mutex是互斥锁,位于System.Threading 命名空间中. 顾名思义,它是一个互斥的对象,同一时间只有一个线程可以拥有它,该类还可用于进程间同步的同步基元. 如果当前有一个线程拥有它 ...
- dubbo入门(一)
1.简介 Dubbo由阿里巴巴开源,是一个分布式服务框架,致力于提供高性能和透明化的RPC(远程过程调用)远程服务调用方案,以及SOA服务治理方案.如果没有分布式的需求,Dbubbo是不需要的,其本质 ...
- java对于Redis中jedis的操作
package com.answer.redis; import java.util.HashMap; import java.util.List; import java.util.Map; imp ...
- 各种对list,string操作函数的总结
#encoding=utf-8#reverse,用来反转lista=['aa','bb','cc']a.reverse()print a#['cc', 'bb', 'aa']#不能直接print a. ...
- 韦大仙--简单的monkey测试命令行操作及生成log日志保存
作中,在将apk交给软件测试人员去测试之前,不免要自己先自测,monkey自测是一个不错的选择! 步骤很简单: 1.测试用的手机与电脑连接好USB ,并且安装好驱动(我一般都是通过豌豆荚自动安装的)! ...
- 游戏AI之群组行为
群组行为指的是多个对象组队同时进行的情况.每个boid需满足分离,队列,凝聚三个基本的规则. 分离:群组中的每个个体都与相邻的个体保持一定的距离. 队列:群组以相同的速度,向相同的方向移动. 凝聚:与 ...
- C 判断成绩是否及格
#include <stdio.h> int main(int argc, char **argv) { // 新建两个变量 pass代表及格分数的固定变量 score代表学生成绩的一个 ...
- TW实习日记:第27天
今天依旧是磨洋工的一天,说真的,被存在各种问题的后端接口把耐心和动力都给磨没了.于是一天就又在沟通接口问题中度过了,完善了一个新功能,将一个新功能开发到了一半.效率可真是够低的,唉.然后不知道为什么突 ...