Android框架之路——GreenDao3.2.2的使用

CSDN同步更新:http://blog.csdn.net/bskfnvjtlyzmv867/article/details/71250101

一、简介

GreenDAO是一个开源的安卓ORM框架,能够使SQLite数据库的开发再次变得有趣。它减轻开发人员处理低级数据库需求,同时节省开发时间。 SQLite是一个令人敬畏的内嵌的关系数据库,编写SQL和解析查询结果是相当乏味和耗时的任务。通过将Java对象映射到数据库表(称为ORM,“对象/关系映射”),GreenDAO可以将它们从这些映射中释放出来,这样,您可以使用简单的面向对象的API来存储,更新,删除和查询数据库。

简单的讲,GreenDAO 是一个将对象映射到 SQLite 数据库中的轻量且快速的 ORM 解决方案。

<p>

</p>

二、ORM概念

对象-关系映射(OBJECT/RELATIONALMAPPING,简称ORM),是随着面向对象的软件开发方法发展而产生的。用来把对象模型表示的对象映射到基于SQL的关系模型数据库结构中去。这样,我们在具体的操作实体对象的时候,就不需要再去和复杂的 SQL 语句打交道,只需简单的操作实体对象的属性和方法。ORM 技术是在对象和关系之间提供了一条桥梁,前台的对象型数据和数据库中的关系型的数据通过这个桥梁来相互转化。

简单的讲,就是JavaBean和我们的数据库进行一个关系映射,一个实例对象对应数据库的一条记录,每个对象的属性则对应着数据库表的字段。

三、添加依赖

// In your root build.gradle file:
buildscript {
    repositories {
        jcenter()
        mavenCentral() // add repository
    }
    dependencies {
        classpath 'com.android.tools.build:gradle:2.3.0'
        classpath 'org.greenrobot:greendao-gradle-plugin:3.2.2' // add plugin
    }
}
 
// In your app projects build.gradle file:
apply plugin: 'com.android.application'
apply plugin: 'org.greenrobot.greendao' // apply plugin
 
dependencies {
    compile 'org.greenrobot:greendao:3.2.2' // add library
}

四、解锁技能

  1. GreenDao 3.0采用注解的方式来定义实体类,通过gradle插件生成相应的代码。您可以使用greenDAO Gradle插件,无需任何其他配置,但至少要设置schema的版本等;

     // In the build.gradle file of your app project:
     android {
     ...
     }
      
     greendao {
         schemaVersion 1
         daoPackage 'com.ping.greendao.gen'
         targetGenDir 'src/main/java'
     }
    

此外,greendao配置元素支持多种配置选项:
- schemaVersion:指定数据库schema版本号,迁移等操作会用到;
- daoPackage:通过gradle插件生成的数据库相关文件的包名,默认为你的entity所在的包名;
- targetGenDir:自定义生成数据库文件的目录,可以将生成的文件放到我们的java目录中,而不是build中,这样就不用额外的设置资源目录了。

  1. 通过GreenDao3注解的语法来定义我们的一个数据库实体类及其数据库操作方法;
    • 我们先生成一个实体类——Meizi,包含id、来源、和图片url地址;

        public class Meizi {
        
            private String _id;
            private String source;
            private String url;
        }
      
    • 通过添加注解为我们的Meizi实体类生成对应的数据库操作方法;

        @Entity
        public class Meizi {
        
            @Id(autoincrement = true)
            private Long _id;
            private String source;
            @NotNull
            private String url;
        }
      

这里的几个注解含义:
- @Entity:将我们的java普通类变为一个能够被greenDAO识别的数据库类型的实体类;
- @nameInDb:在数据库中的名字,如不写则为实体中类名;
- @Id:选择一个long / Long属性作为实体ID。 在数据库方面,它是主键。 参数autoincrement是设置ID值自增;
- @NotNull:使该属性在数据库端成为“NOT NULL”列。 通常使用@NotNull标记原始类型(long,int,short,byte)是有意义的;
- @Transient:表明这个字段不会被写入数据库,只是作为一个普通的java类字段,用来临时存储数据的,不会被持久化。

- 通过点击AndroidStudio中的MakeProject,便发现GreenDao为我们的Meizi实体类生成了对应的Getter、Setter方法以及俩个构造函数,同时在我们配置的com.ping.greendao.gen包下生成了三个对应类文件DaoMaster、DaoSession和MeiziDao,之后所有相关的数据库操作都依靠这三个文件了;

        @Entity
        public class Meizi {
        
            @Id(autoincrement = true)
            private Long _id;
            private String source;
            @NotNull
            private String url;
            @Generated(hash = 717937950)
            public Meizi(Long _id, String source, @NotNull String url) {
                this._id = _id;
                this.source = source;
                this.url = url;
            }
            @Generated(hash = 507723578)
            public Meizi() {
            }
            public Long get_id() {
                return this._id;
            }
            public void set_id(Long _id) {
                this._id = _id;
            }
            public String getSource() {
                return this.source;
            }
            public void setSource(String source) {
                this.source = source;
            }
            public String getUrl() {
                return this.url;
            }
            public void setUrl(String url) {
                this.url = url;
            }
        }

这里要解释一下生成的三个核心类的作用:
<p>

</p>
- DaoMaster:使用greenDAO的切入点。DaoMaster保存数据库对象(SQLiteDatabase)并管理特定模式的DAO类(而不是对象)。 它具有静态方法来创建表或将它们删除。 其内部类OpenHelper和DevOpenHelper是在SQLite数据库中创建模式的SQLiteOpenHelper实现。一个DaoMaster就代表着一个数据库的连接。
- DaoSession:管理特定模式的所有可用DAO对象,您可以使用其中一个getter方法获取。 DaoSession还为实体提供了一些通用的持久性方法,如插入,加载,更新,刷新和删除。 DaoSession可以让我们使用一些Entity的基本操作和获取Dao操作类,DaoSession可以创建多个,每一个都是属于同一个数据库连接的。
- XxxDAO:数据访问对象(DAO)持续存在并查询实体。 对于每个实体,GreenDAO生成一个DAO。 它比DaoSession有更多的持久化方法,例如:count,loadAll和insertInTx。

  1. 进行增删改操作;
    • 编写DaoManager,用于创建数据库、创建数据库表、包含增删改查的操作以及数据库的升级。

       /**
        * 创建数据库、创建数据库表、包含增删改查的操作以及数据库的升级
        * Created by Mr.sorrow on 2017/5/5.
        */
       
       public class DaoManager {
           private static final String TAG = DaoManager.class.getSimpleName();
           private static final String DB_NAME = "greendaotest";
       
           private Context context;
       
           //多线程中要被共享的使用volatile关键字修饰
           private volatile static DaoManager manager = new DaoManager();
           private static DaoMaster sDaoMaster;
           private static DaoMaster.DevOpenHelper sHelper;
           private static DaoSession sDaoSession;
       
           /**
            * 单例模式获得操作数据库对象
            * @return
            */
           public static DaoManager getInstance(){
               return manager;
           }
       
           public void init(Context context){
               this.context = context;
           }
       
           /**
            * 判断是否有存在数据库,如果没有则创建
            * @return
            */
           public DaoMaster getDaoMaster(){
               if(sDaoMaster == null) {
                   DaoMaster.DevOpenHelper helper = new DaoMaster.DevOpenHelper(context, DB_NAME, null);
                   sDaoMaster = new DaoMaster(helper.getWritableDatabase());
               }
               return sDaoMaster;
           }
       
           /**
            * 完成对数据库的添加、删除、修改、查询操作,仅仅是一个接口
            * @return
            */
           public DaoSession getDaoSession(){
               if(sDaoSession == null){
                   if(sDaoMaster == null){
                       sDaoMaster = getDaoMaster();
                   }
                   sDaoSession = sDaoMaster.newSession();
               }
               return sDaoSession;
           }
       
           /**
            * 打开输出日志,默认关闭
            */
           public void setDebug(){
               QueryBuilder.LOG_SQL = true;
               QueryBuilder.LOG_VALUES = true;
           }
       
           /**
            * 关闭所有的操作,数据库开启后,使用完毕要关闭
            */
           public void closeConnection(){
               closeHelper();
               closeDaoSession();
           }
       
           public void closeHelper(){
               if(sHelper != null){
                   sHelper.close();
                   sHelper = null;
               }
           }
       
           public void closeDaoSession(){
               if(sDaoSession != null){
                   sDaoSession.clear();
                   sDaoSession = null;
               }
           }
       }
      
    • 编写XxxDaoUtil,用于完成对某一张数据表的具体操作——ORM操作。以创建MeiziDaoUtil为例:

       public class MeiziDaoUtils {
           private static final String TAG = MeiziDaoUtils.class.getSimpleName();
           private DaoManager mManager;
       
           public MeiziDaoUtils(Context context){
               mManager = DaoManager.getInstance();
               mManager.init(context);
           }
       
           /**
            * 完成meizi记录的插入,如果表未创建,先创建Meizi表
            * @param meizi
            * @return
            */
           public boolean insertMeizi(Meizi meizi){
               boolean flag = false;
               flag = mManager.getDaoSession().getMeiziDao().insert(meizi) == -1 ? false : true;
               Log.i(TAG, "insert Meizi :" + flag + "-->" + meizi.toString());
               return flag;
           }
      
           /**
            * 插入多条数据,在子线程操作
            * @param meiziList
            * @return
            */
           public boolean insertMultMeizi(final List<Meizi> meiziList) {
               boolean flag = false;
               try {
                   mManager.getDaoSession().runInTx(new Runnable() {
                       @Override
                       public void run() {
                           for (Meizi meizi : meiziList) {
                               mManager.getDaoSession().insertOrReplace(meizi);
                           }
                       }
                   });
                   flag = true;
               } catch (Exception e) {
                   e.printStackTrace();
               }
               return flag;
           }
      
           /**
            * 修改一条数据
            * @param meizi
            * @return
            */
           public boolean updateMeizi(Meizi meizi){
               boolean flag = false;
               try {
                   mManager.getDaoSession().update(meizi);
                   flag = true;
               }catch (Exception e){
                   e.printStackTrace();
               }
               return flag;
           }
      
           /**
            * 删除单条记录
            * @param meizi
            * @return
            */
           public boolean deleteMeizi(Meizi meizi){
               boolean flag = false;
               try {
                   //按照id删除
                   mManager.getDaoSession().delete(meizi);
                   flag = true;
               }catch (Exception e){
                   e.printStackTrace();
               }
               return flag;
           }
       
           /**
            * 删除所有记录
            * @return
            */
           public boolean deleteAll(){
               boolean flag = false;
               try {
                   //按照id删除
                   mManager.getDaoSession().deleteAll(Meizi.class);
                   flag = true;
               }catch (Exception e){
                   e.printStackTrace();
               }
               return flag;
           }
      
           /**
            * 查询所有记录
            * @return
            */
           public List<Meizi> queryAllMeizi(){
               return mManager.getDaoSession().loadAll(Meizi.class);
           }
       
           /**
            * 根据主键id查询记录
            * @param key
            * @return
            */
           public Meizi queryMeiziById(long key){
               return mManager.getDaoSession().load(Meizi.class, key);
           }
      
           /**
            * 使用native sql进行查询操作
            */
           public List<Meizi> queryMeiziByNativeSql(String sql, String[] conditions){
               return mManager.getDaoSession().queryRaw(Meizi.class, sql, conditions);
           }
      
           /**
            * 使用queryBuilder进行查询
            * @return
            */
           public List<Meizi> queryMeiziByQueryBuilder(long id){
               QueryBuilder<Meizi> queryBuilder = mManager.getDaoSession().queryBuilder(Meizi.class);
               return queryBuilder.where(MeiziDao.Properties._id.eq(id)).list();
           }
       }
      
    • 单个插入操作:

       case R.id.insert:
          mMeiziDaoUtils.insertMeizi(new Meizi(null, "Google",
                   "http://7xi8d6.48096_n.jpg"));
           break;
      
    • 批量插入操作:

       List<Meizi> meiziList = new ArrayList<>();
        meiziList.add(new Meizi(null, "HuaWei",
                "http://7xi8d648096_n.jpg"));
        meiziList.add(new Meizi(null, "Apple",
                "http://7xi8d648096_n.jpg"));
        meiziList.add(new Meizi(null, "MIUI",
                "http://7xi8d648096_n.jpg"));
        mMeiziDaoUtils.insertMultMeizi(meiziList);
      
    • 单个更改操作:(其中原有的数据都不会保存,如果新建的对象有属性没有设置,则会为空,不为空的字段没有设置,则报错)

       Meizi meizi = new Meizi();
       meizi.set_id(1002l);
       meizi.setUrl("http://baidu.jpg");
       mMeiziDaoUtils.updateMeizi(meizi);
      
    • 删除某条记录操作:

       Meizi meizi1 = new Meizi();
       meizi1.set_id(1002l);
       mMeiziDaoUtils.deleteMeizi(meizi1);
      
    • 删除所有记录操作:

      mMeiziDaoUtils.deleteAll();
      
  2. 专为查询单独列出
    • 查询所有记录:

       case R.id.checksingle:
           Log.i(TAG, mMeiziDaoUtils.queryMeiziById(1008l).toString());
           break;
      
    • 根据主键查询记录:

       case R.id.checkmult:
           List<Meizi> meiziList1 = mMeiziDaoUtils.queryAllMeizi();
           for (Meizi meizi2 : meiziList1) {
               Log.i(TAG, meizi2.toString());
           }
           break;
      
    • 各种条件查询

      • 使用native sql进行条件查询:

         case R.id.queryNativeSql:
             String sql = "where _id > ?";
             String[] condition = new String[]{"1008"};
             List<Meizi> meiziList2 = mMeiziDaoUtils.queryMeiziByNativeSql(sql, condition);
             for (Meizi meizi2 : meiziList2) {
                 Log.i(TAG, meizi2.toString());
             }
             break;
        
      • 使用queryBuilder进行条件查询:
           QueryBuilder能够让你在不涉及SQL语句的情况下查询实体。写SQL有几个缺点,首先是易错的,其次是要在运行时才知道有没有问题(假如属性名是pid,你写成了id,也要到运营时才会崩溃),QueryBuilder能够在编译时检查错误(如属性的引用是否错误)。
           关于Api:在org.greenrobot.greendao.query包下,QueryBuilder类中查看其方法;构造函数可以传递我们的Xxx实体类型,查询方法有很多逻辑的where方法。where方法中需要设置WhereCondition类型的条件参数,而在org.greenrobot.greendao包下的Property类中,每一种操作符的方法都返回WhereCondition类型。获取Property实例则不需要我们去做,在我们的XxxDao中已经有对应的提供,例如我们这里的MeiziDao.Properties.XXX。

         List<Meizi> meiziList2 = mMeiziDaoUtils.queryMeiziByQueryBuilder(1008);
         for (Meizi meizi2 : meiziList2) {
             Log.i(TAG, meizi2.toString());
         }
        

   LazyList懒加载是指一次性查完数据保存在内存中,然后关闭所有连接,再次查询时从内存中获取。一般查询大数据量时用。

五、Demo下载

       源码链接

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,580评论 18 139
  • 一、关于greenDAO greenDAO应该算是当前最火的数据库开源框架了,它是一个将对象映射到SQLite数据...
    当幸福来敲门58阅读 13,842评论 3 19
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,560评论 18 399
  • 类的属性和实例变量是两个概念 类的属性声明通过关键字@property声明, 编译器在编译阶段将自动:1.为属性创...
    羚君阅读 498评论 0 0
  • 雪地飞狐 2016-8-19 10:39 一 老师被学生直呼其名 这种情况,我本人到目前为止还没有遇到过。但见过不...
    雪地飞狐阅读 407评论 2 5