GreenDao2.0使用详解

介绍

GreenDAO是一个对象关系映射(ORM)的框架,能够提供相关接口,通过操作对象的方式去操作关系型数据库。GreenDao官网介绍: greenDAO is a light & fast ORM solution for Android that maps objects to SQLite databases.

greenDao作为java对象和SQLite数据库之间的桥梁

GreenDao的优势
greenDao与OrmLite、ActiveAndroid对比
  1. 轻量级
  2. 增删改查操作快速
  3. 内存开销小,性能好
  4. Api接口完善,方便初学者使用
GreenDao2.0配置

在gradle中引入:

compile 'de.greenrobot:greendao:2.0.0'

GreenDao2.0使用
  • 创建本地数据库及表
    新建一java工程,导入greendao-generator-1.3.1.jar到libs中,main文件中写:
    public class ExampleDaoGenerator {
    public static void main(String[] args) throws Exception {
    //第一个参数是数据库版本号,第二个参数是包名
    Schema schema = new Schema(version, "你的包名路径");
    //给数据库新增一张表
    addMessage(schema);
    //第二个参数是文件生成路径
    new DaoGenerator().generateAll(schema, LocalConstants.CODE_PATH);
    }
    private static void addMessage(Schema schema) {
    //表实体,会对应生成一张表
    Entity note = schema.addEntity("MessageEntity");
    //给表添加一个主键
    note.addStringProperty("msgId").primaryKey().notNull();
    //boolean类型字段,对应列名
    note.addBooleanProperty("isRead");
    //字符类型字段,对应列名
    note.addStringProperty("createTime");
    }
    }
    执行Main()方法,数据库映射文件将会在LocalConstants.CODE_PATH路径下生成。

  • 数据库的使用(表的增删改查)
    public class DbHelper {
    private final static String DB_NAME = "xxx.db";
    private static volatile DbHelper dbHelper;
    private DaoSession daoSession;
    private DaoMaster daoMaster;
    private MessageEntityDao messageEntityDao;
    public static DbHelper getInstance() {
    if (null == dbHelper) {
    synchronized (DbHelper.class) {
    if (null == dbHelper) {
    dbHelper = new DbHelper();
    dbHelper.daoSession = dbHelper.getDaoSession(XxxApplication.getInstance());
    dbHelper.messageEntityDao = dbHelper.daoSession.getMessageEntityDao();
    }
    }
    }
    return dbHelper;
    }
    private DbHelper() {

        }
    
      private DaoSession getDaoSession(Context context) {
        if (daoSession == null) {
          if (daoMaster == null) {
              daoMaster = getDaoMaster(context);
          }
          daoSession = daoMaster.newSession();
        }
        return daoSession;
      }
      private DaoMaster getDaoMaster(Context context) {
      if (daoMaster == null) {
          DaoMaster.OpenHelper helper = new DaoMaster.DevOpenHelper(context, DB_NAME, null);
          daoMaster = new DaoMaster(helper.getWritableDatabase());
      }
      return daoMaster;
    }
    //保存一条消息
    public long saveMessageEntity(MessageEntity messageEntity) {
      if (null == messageEntity) {
          return -1;
      }
      long result = -1;
      try {
          result = messageEntityDao.insertOrReplace(messageEntity);
      } catch (ClassCastException ex) {
          MLog.e("ClassCastException", "ClassCastException");
      } catch (Exception ex) {
          ex.printStackTrace();
      }
      return result;
    }
    //查询未读消息数
    public int getNoReadMsg() {
      try {
          return messageEntityDao.queryBuilder().where(MessageEntityDao.Properties.IsRead.eq(false)).build().list().size();
      } catch (Exception e) {
          e.printStackTrace();
          return 0;
      }
    }
    //查消息类型对应的未读消息条数
    public int getNoReadMsgCountByMsgType(String msgType) {
      try {
          return messageEntityDao.queryBuilder().where( MessageEntityDao.Properties.MsgType.eq(msgType), MessageEntityDao.Properties.IsRead.eq(false)).build().list().size();
      } catch (Exception e) {
          e.printStackTrace();
          return 0;
      }
    }
    //查询消息
    public MessageEntity getMsgByMsgId(String msgId) {
        try {
          return messageEntityDao.queryBuilder().where(MessageEntityDao.Properties.MsgId.eq(msgId)).build().unique();
          } catch (Exception e) {
          e.printStackTrace();
          return null;
          }
      }
    }
    

说明:
使用DevOpenHelper打开数据库
DaoMaster.DevOpenHelper helper = new DevOpenHelper(context,"xxx.db",null);
我们查看下DevOpenHelper内容
public static class DevOpenHelper extends OpenHelper {
public DevOpenHelper(Context context, String name, CursorFactory factory) {
super(context, name, factory);
}

  @Override
  public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
      Log.i("greenDAO", "Upgrading schema from version " + oldVersion + " to " + newVersion + " by dropping all tables");
      dropAllTables(db, true);
      onCreate(db);
  }
  }

可以发现,当数据库版本更新时,会执行onUpgrade方法,先删除原有的数据表,再新建表。

数据库升级

当我们的app版本升级时,当对本地数据库表结构修改时,就有必要对本地数据库升级。我们只需要修改下DaoMaster文件中的SCHEMA_VERSION常量(增1),它就会执行onUpgrade()。
public class DaoMaster extends AbstractDaoMaster {
public static final int SCHEMA_VERSION = 128;
public static void createAllTables(SQLiteDatabase db, boolean ifNotExists) {
MessageEntityDao.createTable(db, ifNotExists);
}
......
}
有时我们并不想把原有的表数据也删除,那要怎么做呢?
其核心思路是

  1. 把旧表改为临时表
  2. 建立新表
  3. 临时表数据写入新表,删除临时表

代码实现:
/**
*表字段有改变的时候需要用到合并数据
*Created by wangfengkai on 17/3/11.
*Github:https://github.com/github/jxwangfengkai
*/
public class DbOpenHelper extends DaoMaster.OpenHelper {
public DbOpenHelper(Context context, String name, SQLiteDatabase.CursorFactory factory) {
super(context, name, factory);
}

    @Override
    public void onUpgrade(SQLiteDatabase sqLiteDatabase, int i, int i1) {
      //操作数据库的更新
      MigrationHelper.getInstance().migrate(sqLiteDatabase, MessageEntityDao.class);
    }
  }

  /**
   * MigrationHelper类
   */
  public class MigrationHelper {
      private static final String CONVERSION_CLASS_NOT_FOUND_EXCEPTION = "MIGRATION HELPER - CLASS DOESN'T MATCH WITH THE CURRENT PARAMETERS";
      private static final String TAG = "MigrationHelper";
      private static volatile MigrationHelper instance;

      public static MigrationHelper getInstance() {
          if (instance == null) {
              instance = new MigrationHelper();
          }
          return instance;
      }

      public void migrate(SQLiteDatabase db, Class<? extends AbstractDao<?, ?>>... daoClasses) {
          generateTempTables(db, daoClasses);
          DaoMaster.dropAllTables(db, true);
          DaoMaster.createAllTables(db, false);
          restoreData(db, daoClasses);
      }

      private void generateTempTables(SQLiteDatabase db, Class<? extends AbstractDao<?, ?>>... daoClasses) {
          for (int i = 0; i < daoClasses.length; i++) {
              DaoConfig daoConfig = new DaoConfig(db, daoClasses[i]);

              String divider = "";
              String tableName = daoConfig.tablename;
              String tempTableName = daoConfig.tablename.concat("_TEMP");
              ArrayList<String> properties = new ArrayList<>();

              StringBuilder createTableStringBuilder = new StringBuilder();

              createTableStringBuilder.append("CREATE TABLE ").append(tempTableName).append(" (");

              for (int j = 0; j < daoConfig.properties.length; j++) {
                  String columnName = daoConfig.properties[j].columnName;

                  if (getColumns(db, tableName).contains(columnName)) {
                      properties.add(columnName);

                      String type = null;

                      try {
                          type = getTypeByClass(daoConfig.properties[j].type);
                      } catch (Exception exception) {
                          Log.e(TAG, "CrashType:" + daoConfig.properties[j].type);
                          exception.printStackTrace();
                      }
                      createTableStringBuilder.append(divider).append(columnName).append(" ").append(type);

                      if (daoConfig.properties[j].primaryKey) {
                          createTableStringBuilder.append(" PRIMARY KEY");
                      }

                      divider = ",";
                  }
              }
              createTableStringBuilder.append(");");
              String createSql = createTableStringBuilder.toString();
              if (!createSql.contains("();")) {
                  //说明没有表
                  db.execSQL(createSql);
                  StringBuilder insertTableStringBuilder = new StringBuilder();

                  insertTableStringBuilder.append("INSERT INTO ").append(tempTableName).append(" (");
                  insertTableStringBuilder.append(TextUtils.join(",", properties));
                  insertTableStringBuilder.append(") SELECT ");
                  insertTableStringBuilder.append(TextUtils.join(",", properties));
                  insertTableStringBuilder.append(" FROM ").append(tableName).append(";");

                  String insertTableSql = insertTableStringBuilder.toString();
                  Log.e(TAG, daoConfig.tablename + "__insertTableSql:" + insertTableSql);
                  db.execSQL(insertTableSql);
              }
          }
      }

      private void restoreData(SQLiteDatabase db, Class<? extends AbstractDao<?, ?>>... daoClasses) {
          for (int i = 0; i < daoClasses.length; i++) {
              DaoConfig daoConfig = new DaoConfig(db, daoClasses[i]);

              String tableName = daoConfig.tablename;
              String tempTableName = daoConfig.tablename.concat("_TEMP");
              ArrayList<String> properties = new ArrayList();

              for (int j = 0; j < daoConfig.properties.length; j++) {
                  String columnName = daoConfig.properties[j].columnName;

                  if (getColumns(db, tempTableName).contains(columnName)) {
                      properties.add(columnName);
                  }
              }

              StringBuilder insertTableStringBuilder = new StringBuilder();

              insertTableStringBuilder.append("INSERT INTO ").append(tableName).append(" (");
              insertTableStringBuilder.append(TextUtils.join(",", properties));
              insertTableStringBuilder.append(") SELECT ");
              insertTableStringBuilder.append(TextUtils.join(",", properties));
              insertTableStringBuilder.append(" FROM ").append(tempTableName).append(";");

              StringBuilder dropTableStringBuilder = new StringBuilder();

              dropTableStringBuilder.append("DROP TABLE IF EXISTS ").append(tempTableName);

              String insertSQL = insertTableStringBuilder.toString();
              if (properties.size() > 0) {
                  Log.e(TAG, daoConfig.tablename + "__restoreData__insertSQL:" + insertSQL);
                  db.execSQL(insertSQL);
              }
              String dropTableSql = dropTableStringBuilder.toString();
              Log.e(TAG, daoConfig.tablename + "__restoreData__dropTableSql:" + dropTableSql);
              db.execSQL(dropTableSql);
          }
      }

      private String getTypeByClass(Class<?> type) throws Exception {
          if (type.equals(String.class)) {
              return "TEXT";
          }
          if (type.equals(Long.class) || type.equals(Integer.class) || type.equals(long.class) || type.equals(int.class) || type.equals(Boolean.class) || type.equals(boolean.class)) {
              return "INTEGER";
          }

          Exception exception = new Exception(CONVERSION_CLASS_NOT_FOUND_EXCEPTION.concat(" - Class: ").concat(type.toString()));
          exception.printStackTrace();
          throw exception;
      }

      private static List<String> getColumns(SQLiteDatabase db, String tableName) {
          List<String> columns = new ArrayList<>();
          Cursor cursor = null;
          try {
              cursor = db.rawQuery("SELECT * FROM " + tableName + " limit 1", null);
              if (cursor != null) {
                  columns = new ArrayList<>(Arrays.asList(cursor.getColumnNames()));
              }
          } catch (Exception e) {
              Log.v(tableName, e.getMessage(), e);
              e.printStackTrace();
          } finally {
              if (cursor != null)
                  cursor.close();
          }
          return columns;
      }
  }

所以,当需要数据库升级时,我们使用
DbOpenHelper helper = new DbOpenHelper(context,"xxx.db",null);
来打开数据库。
讲到这里,GreenDao2.0的使用基本已经完了,下一篇,我将为大家讲解GreenDao3.0的优化及使用。
喜欢就点个赞哦。

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

推荐阅读更多精彩内容

  • http://jinnianshilongnian.iteye.com/blog/2022468 Realm数据...
    天之大任阅读 259评论 0 0
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,100评论 18 139
  • 前言: 上一篇简介了greendao的数据库的接入以及简单的操作,既然涉及到数据库中的数据,那就必须考虑到加密问题...
    Smile__EveryDay阅读 2,190评论 0 8
  • 以前开发用到数据库时,基本上都是用android原生的sql语句,写那些语句时稍有不慎,就给你抛出一个except...
    BlainPeng阅读 7,643评论 2 12
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,296评论 18 399