假如我们的实体类是这样的:
public class Album extends LitePalSupport {
private String name;
private float price;
private byte[] cover;
private Date releaseDate;
private List songs = new ArrayList();
// generated getters and setters.
…
}
此时如果需要更新表结构,只需修改实体类,并更改litepal.xml中的数据库版本号,在下一次操作数据库时就会更新表结构,非常巧妙的设计,那么他究竟是怎么实现的呢?让我们一起去看源码吧。
@Override
public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
Generator.upgrade(db);
SharedUtil.updateVersion(newVersion);
}
这里首先继承了SQLiteOpenHelper类,并重写了onCreate()和onUpgrade()方法。
Generator中的upgrade(SQLiteDatabase db)方法:
static void upgrade(SQLiteDatabase db) {
drop(db);
create(db, false);
updateAssociations(db);
upgradeTables(db);
addAssociation(db, false);
}
第一步删除原数据库,第二步创建新数据库,第三步更新数据库表(移除外键),第四步更新所有数据库中的表,第五步添加外键。各方法的具体实现这里就不赘述了,想了解请移步github。
SharedUtil中的updateVersion(int newVersion)方法:
public static void updateVersion(int newVersion) {
SharedPreferences.Editor sEditor = LitePalApplication.getContext()
.getSharedPreferences(LITEPAL_PREPS, Context.MODE_PRIVATE).edit();
sEditor.putInt(VERSION, newVersion);
sEditor.commit();
}
很明显,这段代码是将新的版本号用SharedPreferences保存在本地。
二、保存数据
使用LitePal保存数据示例:
Album album = new Album();
album.setName(“album”);
album.setPrice(10.99f);
album.setCover(getCoverImageBytes());
album.save();
这个save()方法是继承父类LitePalSupport的方法,想要进行CRUD操作,实体类必须继承LitePalSupport类。现在我们一起来看一下save()方法是怎么实现的:
/*@return If the model is saved successfully, return true. Any exception
-
happens, return false.
*/
public boolean save() {
try {
saveThrows();
return true;
} catch (Exception e) {
e.printStackTrace();
return false;
}
}
savaThrows()方法:
public void saveThrows() {
synchronized (LitePalSupport.class) {
SQLiteDatabase db = Connector.getDatabase();
db.beginTransaction();
try {
SaveHandler saveHandler = new SaveHandler(db);
saveHandler.onSave(this);
clearAssociatedData();
db.setTransactionSuccessful();
} catch (Exception e) {
throw new LitePalSupportException(e.getMessage(), e);
} finally {
db.endTransaction();
}
}
}
这里主要看onSave()方法,这是onSave()方法的关键代码:
String className = baseObj.getClassName();
List supportedFields = getSupportedFields(className);
Collection associationInfos = getAssociationInfo(className);
if (!baseObj.isSaved()) {
analyzeAssociatedModels(baseObj, associationInfos);
doSaveAction(baseObj, supportedFields);
analyzeAssociatedModels(baseObj, associationInfos);
} else {
analyzeAssociatedModels(baseObj, associationInfos);
doUpdateAction(baseObj, supportedFields);
}
如果是已经已经保存的对象则直接执行更新操作,否则执行保存操作。
接下来继续看doSavaAction()方法:
private void doSaveAction(LitePalSupport baseObj, List supportedFields, List supportedGenericFields)
throws SecurityException, IllegalArgumentException, NoSuchMethodException,
IllegalAccessException, InvocationTargetException {
values.clear();
beforeSave(baseObj, supportedFields, values);
long id = saving(baseObj, values);
afterSave(baseObj, supportedFields, supportedGenericFields, id);
}
beforeSave()方法:在保存之前,先对其进行解析,把所有的数据包括字段值和外键值保存在ConteneValues中。saving()方法,返回值是long:
private long saving(LitePalSupport baseObj, ContentValues values) {
if (values.size() == 0) {
values.putNull(“id”);
}
return mDatabase.insert(baseObj.getTableName(), null, values);
}
是不是觉得似曾相识?对的,这里最终执行的还是SQLiteDatabase中的insert方法。同样的doUpdateAction()方法与之类似,最终调用的是SQLiteDatabase中的update方法,这里就不再详叙了。这就是litepal框架保存数据的最终实现。
三、更新数据
保存数据最终是调用SQLiteDatabase中提供的方法实现保存,更新数据会不会也一样呢?接下来咱们一起来看看LitaPal是怎么实现数据更新的。首先看一个使用LitePal更新数据的实例:
Album albumToUpdate = new Album();
albumToUpdate.setPrice(20.99f); // raise the price
albumToUpdate.update(id);
当然更新数据也可以使用sava()方法,原因第二部分讲过了,这里就不再重复。
LitePal提供了update()和updateAll()等方法更新数据,因为实现大同小异,所以我们这里只分析update()方法。
DataSupport.update( id):
public synchronized int update(long id) {
try {
UpdateHandler updateHandler = new UpdateHandler(Connector.getDatabase());
int rowsAffected = updateHandler.onUpdate(this, id);
getFieldsToSetToDefault().clear();
return rowsAffected;
} catch (Exception e) {
throw new DataSupportException(e.getMessage());
}
}
UpdateHandler.onUpdate():
int onUpdate(LitePalSupport baseObj, long id) throws SecurityException, IllegalArgumentException,
NoSuchMethodException, IllegalAccessException, InvocationTargetException {
总结
可以看出,笔者的工作学习模式便是由以下 「六个要点」 组成:
❝ 多层次的工作/学习计划 + 番茄工作法 + 定额工作法 + 批处理 + 多任务并行 + 图层工作法❞
希望大家能将这些要点融入自己的工作学习当中,我相信一定会工作与学习地更富有成效。
下面是我学习用到的一些书籍学习导图,以及系统的学习资料。每一个知识点,都有对应的导图,学习的资料,视频,面试题目。
**如:我需要学习 **Flutter的知识。(大家可以参考我的学习方法)
- Flutter 的思维导图(无论学习什么,有学习路线都会事半功倍)
- Flutter进阶学习全套手册
- Flutter进阶学习全套视频
大概就上面这几个步骤,这样学习不仅高效,而且能系统的学习新的知识。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!
到的一些书籍学习导图,以及系统的学习资料。每一个知识点,都有对应的导图,学习的资料,视频,面试题目。
**如:我需要学习 **Flutter的知识。(大家可以参考我的学习方法)
- Flutter 的思维导图(无论学习什么,有学习路线都会事半功倍)
[外链图片转存中…(img-Ht9KKPyD-1715494824475)]
- Flutter进阶学习全套手册
[外链图片转存中…(img-XcqXC4WE-1715494824476)]
- Flutter进阶学习全套视频
[外链图片转存中…(img-JvYgapg9-1715494824478)]
大概就上面这几个步骤,这样学习不仅高效,而且能系统的学习新的知识。
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门,即可获取!