GreenDao3.0系列文章:
GreenDao3.0 源码分析-DaoMaster和DaoSeesion
图片原图可以从github 源码地址 ->GreenDao 文件中 亿图 打开。
GreenDao3.0源码架构图
上面是我画的GreenDao分层,我们将根据分层逐步分析原理。
GreenDao3.0原理
从图中我们可以看到,其实GreenDao3.0是以Android Sqlite为底层,然后做一些缓存使得大数据的时候能更有效的增删改查。GreenDao最吸引我的:一是对缓存的处理。二就是代码生成。下面我们将一步步对项目源码进行解剖。
GreenDao最底层就是Sqlite 为基础。
工具层:是对Sql语句的处理和Dao配置的解析。
内部类:只要是提供给外层的一些基础类。
对外层:是对客户端的层面,这一层可以直接进行对实体的增删改查。
GreenDao3.0 Helper类
我们先从一开始的出口的帮助类流程开始解析,一步一步引导进来是酱紫的顺便。
DevOpenHelper->OpenHelper->DatabaseOpenHelper->SQLiteOpenHelper
DevOpenHelper核心代码:
![image](http://upload-images.jianshu.io/upload_images/3161886-a1c61c92d22e7ba6?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
他只实现了这个更新的操作,使用这个帮助类,每一次进行版本更新的时候都删除所有的表然后再创建,所以我们每次进行加版本都会导致数据遗失。
OpenHelper核心代码:
![image](http://upload-images.jianshu.io/upload_images/3161886-c6e6f7979bf03040?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
![image](http://upload-images.jianshu.io/upload_images/3161886-af5ed4c64376f709?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
OpenHelper完成了两个功能,一个是建表,一个是传入当前的版本。
DatabaseOpenHelper核心:
DatabaseOpenHelper继承自Sqlite帮助类,通过代理模式的思想,进行狸猫换太子般把SQLiteDatabase->扩展成Database。
![image](http://upload-images.jianshu.io/upload_images/3161886-adeb08e13735dbb9?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
而StandardDatabase实现了Database接口,Database拥有SQLiteDatabase所有的功能方法,只是替换调用,这个类后面再说。
DatabaseOpenHelper第二个功能是扩展了EncryptedHelper继承于net.sqlcipher.database.SQLiteOpenHelper,当我们需要加密的时候可以使用。
总结一下就是:
DatabaseOpenHelper:主要完成SQLiteDatabase->Database和创建加密帮助类两个功能。
OpenHelper:主要完成建表的功能。
DevOpenHelper:主要处理更新数据库版本的时候,进行删库再处理。开发时候建议不使用此类,可以自己重写处理。
Database
上面关系图我们知道,DataBase是GreenDao内的数据库规范接口,StandardDatabase是DataBase的实现,其实就是通过代理模式通过调用SQLiteDatabase完成功能的扩展。
下面我们来说说DataBase各个方法执行的操作:
Cursor rawQuery(String sql, String[] selectionArgs):执行查询的Sql语句返回游标。
void execSQL(String sql) throws SQLException:执行无条件Sql语句。
void execSQL(String sql, Object[] bindArgs) throws SQLException:执行有条件Sql语句。
void beginTransaction()、void endTransaction()、void setTransactionSuccessful():数据库事务三个方法。分别是开启、结束、设置成功。
boolean inTransaction():判断是否在一个事务中。
DatabaseStatement compileStatement(String sql):通过有占位符的Sql语句获取一个SQLiteStatement并包装到DatabaseStatement 。
boolean isDbLockedByCurrentThread():如果当前线程持有到数据库的活动连接,则返回true,此方法的名称来自于与数据库有活动连接的时间,意味着该线程在数据库上持有实际锁定。
Object getRawDatabase():获取源数据库,比如现在使用的代理源数据库是Sqlite,就返回SQLiteDatabase。
DatabaseStatement
DatabaseStatement是SQLiteStatement的代理,解决了SQL注入值的问题,因可以复用优化了性能。提供方法如下。
![image](http://upload-images.jianshu.io/upload_images/3161886-344e911e5288a3f8?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
void execute():绑定了值后,执行sql statement语句。
public long simpleQueryForLong():执行一个sql statement语句,返回一个数值。比如:SELECT COUNT(*) FROM table;
public long executeInsert():执行一个sql statement语句,返回一个rowId.常用于插入。
public void clearBindings():清理绑定。
public Object getRawStatement():获取源Stamtement对象,返回被代理的对象。
public void bindXXX(int index, XX value):对应Index绑定相应的基本类型的值。