转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/39151617
在上一篇文章中,我们学习了LitePal的基本用法,体验了使用框架来进行创建表操作的便利。然而大家都知道,创建表只是数据库操作中最基本的一步而已,我们在一开始创建的表结构,随着需求的变更,到了后期是极有可能需要修改的。因此,升级表的操作对于任何一个项目也是至关重要的,那么今天我们就一起来学习一下,在Android传统开发当中升级表的方式,以及使用LitePal来进行升级表操作的用法。如果你还没有看过前一篇文章,建议先去参考一下Android数据库高手秘籍(二)——创建表和LitePal的基本用法。
传统的升级表方式
上一篇文章中我们借助MySQLiteHelper已经创建好了news这张表,这也是demo.db这个数据库的第一个版本。然而,现在需求发生了变更,我们的软件除了能看新闻之外,还应该允许用户评论,所以这时就需要对数据库进行升级,添加一个comment表。
该怎么做呢?添加一个comment表的建表语句,然后在onCreate()方法中去执行它?没错,这样的话,两张表就会同时创建了,代码如下所示:
-
publicclassMySQLiteHelperextendsSQLiteOpenHelper{
-
-
publicstaticfinalStringCREATE_NEWS="createtablenews("
-
+"idintegerprimarykeyautoincrement,"
-
+"titletext,"
-
+"contenttext,"
-
+"publishdateinteger,"
-
+"commentcountinteger)";
-
-
publicstaticfinalStringCREATE_COMMENT="createtablecomment("
-
+"idintegerprimarykeyautoincrement,"
-
+"contenttext)";
-
-
publicMySQLiteHelper(Contextcontext,Stringname,CursorFactoryfactory,
-
intversion){
-
super(context,name,factory,version);
-
}
-
-
@Override
-
publicvoidonCreate(SQLiteDatabasedb){
-
db.execSQL(CREATE_NEWS);
-
db.execSQL(CREATE_COMMENT);
-
}
-
-
@Override
-
publicvoidonUpgrade(SQLiteDatabasedb,intoldVersion,intnewVersion){
-
}
-
-
}
这对于第一次安装我们软件的用户来说是完全可以正常工作的,但是如果有的用户已经安装过上一版的软件,那么很遗憾,comment表是创建不出来的,因为之前数据库就已经创建过了,onCreate()方法是不会重新执行的。
对于这种情况我们就要用升级的方式来解决了,看到MySQLiteHelper构造方法中的第四个参数了吗,这个就是数据库版本号的标识,每当版本号增加的时候就会调用onUpgrade()方法,我们只需要在这里处理升级表的操作就行了。比较简单粗暴的方式是将数据库中现有的所有表都删除掉,然后重新创建,代码如下所示:
-
publicclassMySQLiteHelperextendsSQLiteOpenHelper{
-
-
......
-
-
@Override
-
publicvoidonCreate(SQLiteDatabasedb){
-
db.execSQL(CREATE_NEWS);
-
db.execSQL(CREATE_COMMENT);
-
}
-
-
@Override
-
publicvoidonUpgrade(SQLiteDatabasedb,intoldVersion,intnewVersion){
-
db.execSQL("droptableifexistsnews");
-
onCreate(db);
-
}
-
-
}
可以看到,当数据库升级的时候,我们先把news表删除掉,然后重新执行了一次onCreate()方法,这样就保证数据库中的表都是最新的了。
但是,如果news表中本来已经有数据了,使用这种方式升级的话,就会导致表中的数据全部丢失,所以这并不是一种值得推荐的升级方法。那么更好的升级方法是什么样的呢?这就稍微有些复杂了,需要在onUpgrade()方法中根据版本号加入具体的升级逻辑,我们来试试来吧。比如之前的数据库版本号是1,那么在onUpgrade()方法中就可以这样写:
-
publicclassMySQLiteHelperextendsSQLiteOpenHelper{
-
-
......
-
-
@Override
-
publicvoidonCreate(SQLiteDatabasedb){
-
db.execSQL(CREATE_NEWS);
-
db.execSQL(CREATE_COMMENT);
-
}
-
-
@Override
-
publicvoidonUpgrade(SQLiteDatabasedb,intoldVersion,intnewVersion){
-
switch(oldVersion){
-
case1:
-
db.execSQL(CREATE_COMMENT);
-
default:
-
}
-
}
-
-
}
可以看到,这里在onUpgrade()方法中加入了一个switch判断,如果oldVersion等于1,就再创建一个comment表。现在只需要调用如下代码,表就可以得到创建或升级了:
-
SQLiteOpenHelperdbHelper=newMySQLiteHelper(this,"demo.db",null,2);
-
SQLiteDatabasedb=dbHelper.getWritableDatabase();
这里我们将版本号加1,如果用户是从旧版本升级过来的,就会新增一个comment表,而如果用户是直接安装的新版本,就会在onCreate()方法中把两个表一起创建了。
OK,现在软件的第二版本也发布出去了,可是就在发布不久之后,突然发现comment表中少了一个字段,我们并没有记录评论发布的时间。没办法,只好在第三版中修复这个问题了,那我们该怎么样去添加这个字段呢?主要需要修改comment表的建表语句,以及onUpgrade()方法中的逻辑,代码如下所示:
-
publicclassMySQLiteHelperextendsSQLiteOpenHelper{
-
-
......
-
-
publicstaticfinalStringCREATE_COMMENT="createtablecomment("
-
+"idintegerprimarykeyautoincrement,"
-
+"contenttext,"
-
+"publishdateinteger)";
-
-
......
-
-
@Override
-
publicvoidonUpgrade(SQLiteDatabasedb,intoldVersion,intnewVersion){
-
switch(oldVersion){
-
case1:
-
db.execSQL(CREATE_COMMENT);
-
case2:
-
db.execSQL("altertablecommentaddcolumnpublishdateinteger");
-
default:
-
}
-
}
-
-
}
可以看到,在建表语句当中我们新增了publishdate这一列,这样当执行onCreate()方法去创建表的时候,comment表中就会有这一列了。那么如果是从旧版本升级过来的呢?也没有问题,我们在onUpgrade()方法中已经把升级逻辑都处理好了,当oldVersion等于2的时候,会执行alter语句来添加publishdate这一列。现在调用以下代码来创建或升级数据库:
-
SQLiteOpenHelperdbHelper=newMySQLiteHelper(this,"demo.db",null,3);
-
SQLiteDatabasedb=dbHelper.getWritableDatabase();
将数据库版本号设置成3,这样就可以保证数据库中的表又是最新的了。
这里我们要注意一个细节,switch中每一个case的最后都是没有使用break的,为什么要这么做呢?这是为了保证在跨版本升级的时候,每一次的数据库修改都能被全部执行到。比如用户当前是从第二版程序升级到第三版程序的,那么case 2中的逻辑就会执行。而如果用户是直接从第一版程序升级到第三版程序的,那么case 1和case 2中的逻辑都会执行。使用这种方式来维护数据库的升级,不管版本怎样更新,都可以保证数据库的表结构是最新的,而且表中的数据也完全不会丢失了。
现在我们已经学习了新增表和新增列这两种升级方式,那么如果是某张表中的某一列已经没有用了,我想把这一列删除掉该怎么写呢?很遗憾,SQLite并不支持删除列的功能,对于这情况,多数软件采取的作法是无视它,反正以后也用不到它了,留着也占不了什么空间,所以针对于这种需求,确实没什么简单的解决办法。
这大概就是传统开发当中升级数据库表的方式了,虽说能写出这样的代码表示你已经对数据库的升级操作理解的比较清楚了,但随着版本越来越多,onUpgrade()方法中的逻辑也会变得愈发复杂,稍微一不留神,也许就会产生错误。因此,如果能让代码自动控制升级逻辑,而不是由人工来管理,那就是再好不过了,那么下面我们就来学习一下怎样使用LitePal来进行升级表的操作。
使用LitePal升级表
通过上一篇文章的学习,我们已经知道LitePal是一款ORM模式的框架了,已经熟悉创建表流程的你,相信对于升级表也一定会轻车熟路的。那么为了模仿传统升级表方式中的需求,现在我们也需要创建一张comment表。第一步该怎么办呢?相信你也许早就已经猜到了,那当然是先创建一个Comment类了,如下所示:
-
packagecom.example.databasetest.model;
-
-
publicclassComment{
-
-
privateintid;
-
-
privateStringcontent;
-
-
-
...
-
}
OK,Comment类中有id和content这两个字段,也就意味着comment表中会有id和content这两列。
接着修改litepal.xml中的配置,在映射列表中新增Cooment类,并将版本号加1,如下所示:
-
<?xmlversion="1.0"encoding="utf-8"?>
-
<litepal>
-
<dbnamevalue="demo"></dbname>
-
-
<versionvalue="2"></version>
-
-
<list>
-
<mappingclass="com.example.databasetest.model.News"></mapping>
-
<mappingclass="com.example.databasetest.model.Comment"></mapping>
-
</list>
-
</litepal>
没错,就是这么简单,仅仅两步,升级的操作就已经完成了,现在我们只需要操作一下数据库,comment表就会自动生成了,如下所示:
-
SQLiteDatabasedb=Connector.getDatabase();
那么我们还是通过.table命令来查看一下结果,如下图所示:
OK,comment表已经出来了,那么再通过pragma命令来查看一下它的表结构吧:
没有问题,comment表中目前有id和content这两列,和Comment模型类中的字段是保持一致的。
那么现在又来了新的需求,需要在comment表中添加一个publishdate列,该怎么办呢?不用怀疑,跟着你的直觉走,相信你已经猜到应该在Comment类中添加这样一个字段了吧,如下所示:
-
publicclassComment{
-
-
privateintid;
-
-
privateStringcontent;
-
-
privateDatepublishDate;
-
-
-
...
-
}
然后呢?剩下的操作已经非常简单了,只需要在litepal.xml中对版本号加1就行了,如下所示:
-
<litepal>
-
<dbnamevalue="demo"></dbname>
-
-
<versionvalue="3"></version>
-
...
-
</litepal>
这样当我们下一次操作数据库的时候,publishdate列就应该会自动添加到comment表中。调用Connector.getDatabase()方法,然后重新查询comment表结构,如下所示:
可以看到,publishdate这一列确实已经成功添加到comment表中了。
通过这两种升级方式的对比,相信你已经充分体会到了使用LitePal进行升级表操作所带来的便利了吧。我们不需要去编写任何与升级相关的逻辑,也不需要关心程序是从哪个版本升级过来的,唯一要做的就是确定好最新的Model结构是什么样的,然后将litepal.xml中的版本号加1,所有的升级逻辑就都会自动完成了。LitePal确实将数据库表的升级操作变得极度简单,使很多程序员可以从维护数据库表升级的困扰中解脱出来。
然而,LitePal却明显做到了更好。前面我们提到过关于删除列的问题,最终的结论是无法解决,因为SQLite是不支持删除列的命令的。但是如果使用LitePal,这一问题就可以简单地解决掉,比如说publishdate这一列我们又不想要了,那么只需要在Comment类中把它删除掉,然后将版本号加1,下次操作数据库的时候这个列就会不见了。
那么有的朋友可能会问了,不是说SQLite不支持删除列的命令吗?那LitePal又是怎样做到的呢?其实LitePal并没有删除任何一列,它只是先将comment表重命名成一个临时表,然后根据最新的Comment类的结构生成一个新的comment表,再把临时表中除了publishdate之外的数据复制到新的表中,最后把临时表删掉。因此,看上去的效果好像是做到了删除列的功能。
这也是使用框架的好处,如果没有框架的帮助,我们显然不会为了删除一个列而大废周章地去写这么多的代码,而使用框架的话,具体的实现逻辑我们已经不用再关心,只需要控制好模型类的数据结构就可以了。
另外,如果你想删除某一张表的话,操作也很简单,在litepal.xml中的映射列表中将相应的类删除,表自然也就不存在了。其它的一些升级操作也都是类似的,相信你已经能举一反三,这里就不再赘述了。
好了,今天对LitePal的介绍就到这里吧,下篇文章当中我们会学习使用LitePal来进行表关联的操作。
分享到:
相关推荐
需要考虑这样的问题:因为我们开发的软件可能会安装在很多用户的手机上,如果应用使用到了SQLite数据库,我们必须在用户初次使用软件时创建出应用使用到的数据库表结构及添加一些初始化记录,另外在软件升级的时候,...
LitePal是一个开源的Android库,它允许开发人员非常容易地使用SQLite数据库。您可以在不编写SQL语句的情况下完成大多数数据库操作,包括创建或升级表、crud操作、聚合函数等。LitePal的设置也很简单,您可以在5分钟...
SQLite是一种超轻量级的嵌入式数据库,大小只有几百KB,但是其语法支持标准SQL语法,同时还遵循了数据库的ACID事务,所以学过其他数据库的开发人员都很容易掌握其使用。 sql语法就不介绍了,直接看在android中的使用...
Android应用源码之sqlite的一些基本操作,包括数据库创建、数据库版本升级、创建表、数据的增删改查.zip项目安卓应用源Android应用源码之sqlite的一些基本操作,包括数据库创建、数据库版本升级、创建表、数据的增删...
嵌入式Android项目设计与开发 第六章 数据存储 ——SQLite 1 SQLite 概念: SQLite是一款轻量级的关系型数据库,它的运算效率高,占用资源少,因此比较适合在移动设备上使用。 SQLite Android为了让我们更加方便地...
Android应用源码开发Demo,主要用于毕业设计学习。
SQLiteDatabase 是 Android 中操作数据库的核心类之一,使用SQLiteDatabase可以打开数据库,也可以对数据库进行操作,然而,为了数据库升级以及使用更加方便,我们常用SQLiteOpenHelper的子类来完成创建,打开数据库...
android APP的开发并不是一次就完美,而是通过多次的升级,升级的过程中会遇到修改app的数据库,这是需要将旧版本的app的数据库复制一份,进行操作。
LitePal是一个开放源代码的Android库,使开发人员可以非常轻松地使用SQLite数据库。 您无需编写SQL语句即可完成大多数数据库操作,包括创建或升级表,crud操作,聚合函数等。LitePal的设置也非常简单,您可以在不到5...
传智播客_Andorid教程_20天从零到精通精通Android开发视频_第05天 1、SQlite数据库简介 2、SQliteOpenHelper介绍获取 3、数据库升级 4、直接执行sql操作数据库,增删改查 5、通过谷歌封装的api操作数据库,增删改查 ...
Android为了让我们更加方便的管理数据库,专门提供了SQLiteOpenHelper帮助类,借助这个类可以简单的对数据库进行创建和升级。 下面介绍一下SQLiteOpenHelper:(扩展知识) SQLiteOpenHelper类介绍 定义:...
Android应用源码,安卓学习设计应用源代码,可供学习及设计参考。
需要考虑这样的问题:因为我们开发的软件可能会安装在很多用户的手机上,如果应用使用到了SQLite数据库,我们必须在用户初次使用软件时创建出应用使用到的数据库表结构及添加一些初始化记录,另外在软件升级的时候,...
LitePal是一个开放源代码的Android库,使开发人员可以非常轻松地使用SQLite数据库。 您无需编写SQL语句即可完成大多数数据库操作,包括创建或升级表,crud操作,聚合函数等。LitePal的设置也非常简单,您可以在不到5...
Android Android Android Android 开发背景 � 计算技术、无线接入技术的发展,使嵌入式系统逐渐有能力对桌面系统常规业务进行支持。 � 谷歌长期以来奉行的移动发展战略:通过与全球各地的手机制造商和移动运营商...
sqlite的一些基本操作,包括数据库创建、数据库版本升级、创建表、数据的增删改查.z项目安卓应用源码下载ipsqlite的一些基本操作,包括数据库创建、数据库版本升级、创建表、数据的增删改查.z项目安卓应用源码下载ip...
本文实例讲述了Android创建或升级数据库时执行的语句,如果是创建或升级数据库,请使用带List参数的构造方法,带SQL语句的构造方法将在数据库创建或升级时执行。 具体程序代码如下: import java.util.List; import...
本系统是一个手机客户端,它主要应用android开发技术,以SQLite作为本地数据库,管理本地数据;使用Activity、Service等组件,并遵照MVC模式进行开发。 通过对Android的初中生体育私教平台软件系统的设计与实现,...