热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

关于Android数据库升级的实践(以ormlite为例)

在我们的安卓开发中,难免会使用到本地存储,而android本地数据库就是一样比较好,能更灵活存储和查询本地数据,但是随着需求的不断变化,难免要对本地数据库的表或者表的某个字段进行操作,而又不想破坏已经

在我们的安卓开发中,难免会使用到本地存储,而android本地数据库就是一样比较好,能更灵活存储和查询本地数据,但是随着需求的不断变化,难免要对本地数据库的表或者表的某个字段进行操作,而又不想破坏已经存储在安卓手机里的旧数据,那么怎么办呢?呵呵,安卓数据库已经帮我们想到了,就是本地数据库升级,当然也可以数据库降级,类似吧。

Android 本地数据库又叫SQLite,它有自己的一些原则:

1.      我们只能重命名据据库中的表名和向旧表中增加新的字段

2.      Android版SQLite不支持修改表中的字段名和删除表中的字段,也不允许改变表中字段的约束条件(但是修改也可以通过增加来实现,我这里叫它打引号的’修改’)

3.      Android版SQLite在存储字段数据时不分类型,所以可以改变字段的类型而无需要升级

 (参考地址:http://ormlite.com/javadoc/ormlite-core/doc-files/ormlite_4.html#Upgrading-Schema)

 

下面就让我使用例子来实践数据库的中字段的增加和’修改’字段吧

下面就让我使用例子来实践数据库的中字段的增加和’修改’字段吧

首先就去下载数据库工具的jar包,我找了一个地址:http://www.java2s.com/Code/Jar/o/Downloadormliteandroid418javadocjar.htm

当然也可以去看看他的源码:https://github.com/j256


看看我们的DatabaseHelper.java

public class DatabaseHelper extends OrmLiteSqliteOpenHelper {
private static final String TABLE_NAME = "upgrade_example.db";
public final static int version = 3; // 数据库版本
private Map daos = new HashMap();
/**
* userDao ,每张表对于一个
*/
private Dao userDao;

private DatabaseHelper(Context context){
super(context, TABLE_NAME, null, version);
}

@Override
public void onCreate(SQLiteDatabase database,
ConnectionSource connectionSource) {
Log.e(DatabaseHelper.class.getName(), "DatabaseHelper onCreate() is invoked...");
try {
TableUtils.createTable(connectionSource, UserInfo.class);
} catch (SQLException e) {
e.printStackTrace();
}
}

@Override
public void onUpgrade(SQLiteDatabase database,ConnectionSource connectionSource, int oldVersion, int newVersion) {
Log.e(DatabaseHelper.class.getName(), "DatabaseHelper onUpgrade() oldVersion:"+oldVersion+",newVersion:"+newVersion);
if(oldVersion<2){
try {
getUserDao().executeRaw("ALTER TABLE `tb_user_info` ADD COLUMN age INTEGER DEFAULT 0;");
} catch (SQLException e) {
e.printStackTrace();
}
}

if(oldVersion<3){
try {
getUserDao().executeRaw("ALTER TABLE `tb_user_info` ADD COLUMN addr TEXT DEFAULT 'china';");
getUserDao().executeRaw("ALTER TABLE `tb_user_info` ADD COLUMN sex TEXT DEFAULT '男';");
getUserDao().updateRaw("UPDATE `tb_user_info` SET sex = '女' WHERE gender = 1;");
} catch (SQLException e) {
e.printStackTrace();
}
}
}

private static DatabaseHelper instance;

/**
* 单例获取该Helper
*
* @param context
* @return
*/
public static synchronized DatabaseHelper getHelper(Context context) {
if (instance == null) {
synchronized (DatabaseHelper.class) {
if (instance == null)
instance = new DatabaseHelper(context);
}
}

return instance;
}

public synchronized Dao getDao(Class clazz) throws SQLException {
Dao dao = null;
String className = clazz.getSimpleName();
if (daos.containsKey(className)) {
dao = daos.get(className);
}
if (dao == null) {
dao = super.getDao(clazz);
daos.put(className, dao);
}
return dao;
}

/**
* 获得userDao
*
* @return
* @throws SQLException
*/
public Dao getUserDao() throws SQLException{
if (userDao == null){
userDao = getDao(UserInfo.class);
}
return userDao;
}

/**
* 释放资源
*/
@Override
public void close(){
super.close();
userDao = null;
}

}
以用户为例的UserInfo.java

@DatabaseTable(tableName = "tb_user_info")
public class UserInfo {
@DatabaseField
private int id;//id
@DatabaseField
private String name;//名称
@DatabaseField
private int gender;//性别 0:男,1:女


public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getGender() {
return gender;
}
public void setGender(int gender) {
this.gender = gender;
}


/*@Override
public String toString() {//database version=1
return "UserInfo [id=" + id + ", name=" + name + ",gender="+gender
+ "]";
}*/


// database version=2 added
@DatabaseField
private int age;//age
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
/*@Override
public String toString() {//database version=2
return "UserInfo [id=" + id + ", name=" + name+ ", age=" + age + ",gender="+gender
+ "]";
}*/



//database version=3 added
@DatabaseField
private String sex;//修改字段
@DatabaseField
private String addr;//增加字段

public String getSex() {
return sex;
}
public void setSex(String sex) {
this.sex = sex;
}
public String getAddr() {
return addr;
}
public void setAddr(String addr) {
this.addr = addr;
}
@Override
public String toString() {
return "UserInfo [id=" + id + ", name=" + name + ", age=" + age+",sex="+sex+",addr="+addr
+ "]";
}
}
这两个主要文件贴出来了,那么我再把主界面的初始化的代码贴一下:

private void initDatabaseDefaultVal(){
SharedPreferences prefs=PreferenceManager.getDefaultSharedPreferences(this);
boolean isHadInitDatabase=prefs.getBoolean("isHadInitDatabase", false);
if(!isHadInitDatabase){
ArrayList userList=new ArrayList();
UserInfo user=new UserInfo();
user.setId(1);
user.setName("李明");
user.setGender(0);
userList.add(user);

user=new UserInfo();
user.setId(2);
user.setName("张婷");
user.setGender(1);
userList.add(user);

user=new UserInfo();
user.setId(3);
user.setName("王强");
user.setGender(0);
userList.add(user);
try {
DatabaseHelper.getHelper(this).getUserDao().create(userList);

Editor editor=prefs.edit();
editor.putBoolean("isHadInitDatabase", true);
editor.commit();
} catch (SQLException e) {
e.printStackTrace();
Log.e(this.getClass().getName(), "init database occupt SQLException!");
}

}
}

好了,我就讲讲我的升级过程(如下图):

一开始我在tb_user_info这张表中,只有id,name,gender这三个字段,这时的数据库版本为1

打印出来的日志如下图

现在升级为版本2,在UserInfo里面增加age(随便把toString()也变一下),把DatabaseHelper的version设置为2,在DatabaseHelper增加onUpgrade函数,写上如下代码:

if(oldVersion<2){
try {
getUserDao().executeRaw("ALTER TABLE `tb_user_info` ADD COLUMN age INTEGER DEFAULT 0;");
} catch (SQLException e) {
e.printStackTrace();
}
}

运行结果如下图:

现在升级为版本3,在UserInfo里面增加addr和sex(随便把toString()也变一下),把DatabaseHelper的version设置为3,在DatabaseHelper的onUpgrade函数增加上如下代码:

if(oldVersion<3){
try {
getUserDao().executeRaw("ALTER TABLE `tb_user_info` ADD COLUMN addr TEXT DEFAULT 'china';");
getUserDao().executeRaw("ALTER TABLE `tb_user_info` ADD COLUMN sex TEXT DEFAULT '男';");
getUserDao().updateRaw("UPDATE `tb_user_info` SET sex = '女' WHERE gender = 1;");
} catch (SQLException e) {
e.printStackTrace();
}
}
运行结果如下图:

当然也可以从版本1直接长到版本3,效果如下图:


好了,实践已经结束,我会把源代码上传上来,地址:http://download.csdn.net/detail/bright789/9447167




推荐阅读
  • MyBatis多表查询与动态SQL使用
    本文介绍了MyBatis多表查询与动态SQL的使用方法,包括一对一查询和一对多查询。同时还介绍了动态SQL的使用,包括if标签、trim标签、where标签、set标签和foreach标签的用法。文章还提供了相关的配置信息和示例代码。 ... [详细]
  • 本文介绍了使用kotlin实现动画效果的方法,包括上下移动、放大缩小、旋转等功能。通过代码示例演示了如何使用ObjectAnimator和AnimatorSet来实现动画效果,并提供了实现抖动效果的代码。同时还介绍了如何使用translationY和translationX来实现上下和左右移动的效果。最后还提供了一个anim_small.xml文件的代码示例,可以用来实现放大缩小的效果。 ... [详细]
  • PDO MySQL
    PDOMySQL如果文章有成千上万篇,该怎样保存?数据保存有多种方式,比如单机文件、单机数据库(SQLite)、网络数据库(MySQL、MariaDB)等等。根据项目来选择,做We ... [详细]
  • Whatsthedifferencebetweento_aandto_ary?to_a和to_ary有什么区别? ... [详细]
  • http头_http头部注入
    1、http头部注入分析1、原理 ... [详细]
  • 本文介绍了数据库的存储结构及其重要性,强调了关系数据库范例中将逻辑存储与物理存储分开的必要性。通过逻辑结构和物理结构的分离,可以实现对物理存储的重新组织和数据库的迁移,而应用程序不会察觉到任何更改。文章还展示了Oracle数据库的逻辑结构和物理结构,并介绍了表空间的概念和作用。 ... [详细]
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • 如何使用Java获取服务器硬件信息和磁盘负载率
    本文介绍了使用Java编程语言获取服务器硬件信息和磁盘负载率的方法。首先在远程服务器上搭建一个支持服务端语言的HTTP服务,并获取服务器的磁盘信息,并将结果输出。然后在本地使用JS编写一个AJAX脚本,远程请求服务端的程序,得到结果并展示给用户。其中还介绍了如何提取硬盘序列号的方法。 ... [详细]
  • Oracle seg,V$TEMPSEG_USAGE与Oracle排序的关系及使用方法
    本文介绍了Oracle seg,V$TEMPSEG_USAGE与Oracle排序之间的关系,V$TEMPSEG_USAGE是V_$SORT_USAGE的同义词,通过查询dba_objects和dba_synonyms视图可以了解到它们的详细信息。同时,还探讨了V$TEMPSEG_USAGE的使用方法。 ... [详细]
  • 在Xamarin XAML语言中如何在页面级别构建ControlTemplate控件模板
    本文介绍了在Xamarin XAML语言中如何在页面级别构建ControlTemplate控件模板的方法和步骤,包括将ResourceDictionary添加到页面中以及在ResourceDictionary中实现模板的构建。通过本文的阅读,读者可以了解到在Xamarin XAML语言中构建控件模板的具体操作步骤和语法形式。 ... [详细]
  • 本文讨论了如何使用Web.Config进行自定义配置节的配置转换。作者提到,他将msbuild设置为详细模式,但转换却忽略了带有替换转换的自定义部分的存在。 ... [详细]
  • Activiti7流程定义开发笔记
    本文介绍了Activiti7流程定义的开发笔记,包括流程定义的概念、使用activiti-explorer和activiti-eclipse-designer进行建模的方式,以及生成流程图的方法。还介绍了流程定义部署的概念和步骤,包括将bpmn和png文件添加部署到activiti数据库中的方法,以及使用ZIP包进行部署的方式。同时还提到了activiti.cfg.xml文件的作用。 ... [详细]
  • 本文介绍了在Ubuntu系统中清理残余配置文件和无用内容的方法,包括清理残余配置文件、清理下载缓存包、清理不再需要的包、清理无用的语言文件和清理无用的翻译内容。通过这些清理操作可以节省硬盘空间,提高系统的运行效率。 ... [详细]
  • 本文提供了关于数据库设计的建议和注意事项,包括字段类型选择、命名规则、日期的加入、索引的使用、主键的选择、NULL处理、网络带宽消耗的减少、事务粒度的控制等方面的建议。同时还介绍了使用Window Functions进行数据处理的方法。通过遵循这些建议,可以提高数据库的性能和可维护性。 ... [详细]
  • 开发笔记:(002)spring容器中bean初始化销毁时执行的方法及其3种实现方式
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了(002)spring容器中bean初始化销毁时执行的方法及其3种实现方式相关的知识,希望对你有一定的参考价值。 ... [详细]
author-avatar
白变小樱啊
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有