本文主要介绍关系型数据库 SQLite 和对象关系映射工具 ORMLite ,两种方式都可以满足Android多数据持久化存储的需求。
- SQLite
简介:SQLite是一款Android内置的轻量级的关系型数据库,它的运算速度非常快,占用资源少,通常只需要几百K的内存就足够。SQLite不仅支持标准的SQL语法,还遵循了数据库的ACID事务(ACID,是指在可靠数据库管理系统(DBMS)中,事务(Transaction)所应该具有的四个特性:原子性(Atomicity)、一致性(Consistency)、隔离性(Isolation)、持久性(Durability))。
使用:首先创建数据库我们创建MydataBaseHelper
类并继承SQLiteOpenHelper(SQLiteOpenHelper是一个抽象类,意味着我们要使用它的话就需要创建一个类去继承它并实现其中的方法,分别是onCreate( )和onUpgrade( ))
MydataBaseHelper.java
public class MyDatabaseHelper extends SQLiteOpenHelper {
//用于创建Book表
public static final String CREATE_BOOK = "create table Book ("
+ "id integer primary key autoincrement, "
+ "author text, "
+ "price real, "
+ "pages integer, "
+ "name text, "
+ "catagory_id integer)";
//用于创建Category表
public static final String CREATE_CATEGORY = "create table Category ("
+ "id integer primary key autoincrement, "
+ "category_name text, "
+ "category_code integer)";
private Context mContext;
public MyDatabaseHelper(Context context, String name,
SQLiteDatabase.CursorFactory factory, int version) {
super(context, name, factory, version);
mContext = context;
}
//第一次创建数据库时会调用此方法
@Override
public void onCreate(SQLiteDatabase db) {
db.execSQL(CREATE_BOOK);
db.execSQL(CREATE_CATEGORY);
}
//数据库版本升级时会调用此方法
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
/*第一版的数据库只有Book表,第二版的数据库增加了Category表,
* 为了保证用户体验,在不干扰前一版的数据的情况下,实现对数据
* 库的平滑升级,简单的可以用此方法进行判断在升级*/
switch (oldVersion){
case 1:
db.execSQL(CREATE_CATEGORY);
case 2:
db.execSQL("alter table Book add column category_id integer");
default:
}
}
}
在MainActivity中创建MyDatabaseHelper对象并调用getReadableDatabase( )或者进行数据库的创建getWritableDatabase( ) ,(两者不同点在于,当数据库不可写入时(如磁盘空间已满)getReadableDatabase( )方法返回的对象将以只读的方式去打开数据库,而getWritableDatabase( )方法则将出现异常);对于数据库的操作完成后都要database.close();防止内存泄漏;
//数据库文件名称
private static final String DATABASE_NAME = "BookStore.db";
//数据库版本号private static final int DATABASE_VERSION = 3;
private MyDatabaseHelper mMyDatabaseHelper;
mMyDatabaseHelper = new MyDatabaseHelper(this, DATABASE_NAME, null, DATABASE_VERSION);
mMyDatabaseHelper.getReadableDatabase();
- 对数据库的增删改查
向数据库添加数据
SQLiteDatabase db = mMyDatabaseHelper.getReadableDatabase();
ContentValues values = new ContentValues();
values.put("name", "The Da Vinci Code");
values.put("author", "Dan Brown");
values.put("pages", 454);
values.put("price", 16.96);
//向数据库插入数据
//**nullColumnHack**
//当values参数为空或者里面没有内容的时候,insert是会失败的(底层[数据库]//(http://lib.csdn.net/base/mysql)不允许插入一个空行),为了防止这种情况,要在这里指定一个列名,到时候如果发现将要插入的行为空行时,就会将你指定的这个列名的值设为null,然后再向数据库中插入。通过观察源码的insertWithOnConflict方法可以看到当ContentValues类型的数据initialValues为null或size<=0时,就会在sql语句中添加nullColumnHack的设置。
若不添加nullColumnHack则sql语句最终的结果将会类似insert into tableName()values();这是不允许的。
若添加上nullColumnHack则sql语句将会变成insert into tableName (nullColumnHack)values(null);这是可以的。
db.insert("Book", null, values);
values.clear();
values.put("name", "The Lost Symbol");
values.put("author", "Dan Brown");
values.put("pages", 510);
values.put("price", 19.96);
db.insert("Book", null, values);
更新数据库的数据
SQLiteDatabase db = mMyDatabaseHelper.getReadableDatabase();
ContentValues values = new ContentValues();
values.put("price", 10.56);
//更新数据库的数据
db.update("Book", values, "name=?", new String[]{"The Da Vinci Code"});
删除数据库的数据
SQLiteDatabase db = mMyDatabaseHelper.getReadableDatabase();
//删除数据库的数据
db.delete("Book", "pages > ?", new String[]{"500"});
查询数据(对数据库的查询操作有很多种方式,这里只给出常见的一种)
SQLiteDatabase db = mMyDatabaseHelper.getReadableDatabase();
//对数据库表进行查询,会返回游标
Cursor cursor = db.query("Book", null, null, null, null, null, null);
while (cursor.moveToNext()){
String name = cursor.getString(cursor.getColumnIndex("name"));
String author = cursor.getString(cursor.getColumnIndex("author"));
int pages = cursor.getInt(cursor.getColumnIndex("pages"));
double price = cursor.getDouble(cursor.getColumnIndex("price"));
Log.d("Query BookStore.db", name);
Log.d("Query BookStore.db", author);
Log.d("Query BookStore.db", pages+"");
Log.d("Query BookStore.db", price+"");
}
cursor.close();
数据库的事务操作
- 模拟一个应用场景:进行一次转账操作,银行会将转账的金额先从你的账户中扣除,然后再向收款方的账户中添加等量的金额。看上去好像没有什么问题,可是当你的账户的金额刚刚被扣除,这是由于一些异常原因导致对方收款失败(比如突然断电),这一部分钱就凭空消失了,当然银行自然会考虑到这个问题,它会保证扣钱和收款的操作要么一起完成,要么都不会成功,而使用的技术就是事物了。
SQLiteDatabase db = mMyDatabaseHelper.getReadableDatabase();
//开启事务
db.beginTransaction();
try {
db.delete("Book", null, null);
Toast.makeText(MainActivity.this, 删除数据成功",Toast.LENGTH_SHORT).show();
/*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.15);
db.insert("Book", null, values);
//事务已经执行成功
db.setTransactionSuccessful();
Toast.makeText(MainActivity.this, "插入数据成功", Toast.LENGTH_SHORT).show();
} catch (NullPointerException e) {
e.printStackTrace();
} finally {
//结束事务
db.endTransaction();
}
- ORMLite
简介:
首先可以去它的官网看看www.ormlite.com,它的英文全称是Object Relational Mapping,意思是对象关系映射;如果接触过Java EE开发的,一定知道Java Web开发就有一个类似的数据库映射框架——Hibernate。简单来说,就是我们定义一个实体类,利用这个框架,它可以帮我们吧这个实体映射到我们的数据库中,在Android中是SQLite,数据中的字段就是我们定义实体的成员变量。
使用:
-
下载 ORMLite Jar首先去ORMLite官网下载jar包,对于Android为:ormlite-android-4.48.jar
和 ormlite-core-4.48.jar
具体可从参考:http://www.jianshu.com/p/05782b598cf0