最近公司项目选用GreenDao作为Android客户端本地数据库的对象关系映射框架。对于GreenDao虽然以往也有简单用过,但这还是笔者第一次在实际业务中使用。碰到了题目所述的两个问题,虽然在Tutorial里和百度没找到答案,但在官方issue里搜了一圈果然有方案,遂记录下来帮助更多人。

综合主键

需求场景:某张表里需要两个或多个column组合在一起成为一个综合主键。比如你的表里需要存储一个用户的账单,虽然账单也有id,但是你希望一张表存储所有用户,那么就需要把userId和账单id放在一起作为一个综合主键。虽然sqlite数据库是可以直接指定综合主键的,但不幸的是GreenDao并不支持(笔者也很懵逼为啥不支持)。

查看官方方案来看一下:

@Entity(      // Define indexes spanning multiple columns here.indexes = {@Index(value = "prop1 DESC, prop2 DESC", unique = true)}
)
public class YourEntity {@Id(autoincrement = true) Long id; Integer prop1; Long prop2; ... } 

可以看到,它是通过一个加上unique约束的综合索引来间接实现综合主键的需求。这样的话还是要用@Id注解定义一个主键,但是我们完全不用管理这个主键,后面插入数据的适合这个字段直接插入null即可。真正的主键是prop1,prop2这两个组合的综合主键,使用insertOrReplace方法插入数据时,如果这两个值完全相等的话,就会替换原有数据。不放心的话大家可以试一试,笔者已经试过了。

存储自定义类型对象

需求场景:sqlite数据库只能直接存储数字、字符串、日期等简单类型,如果要存储一个复杂对象的话需要把对象拆解为一个个简单数据类型,毕竟再复杂的数据类型也是由简单数据类型组合而来。本以为大名鼎鼎的GreenDao可以帮我们智能拆解、组装对象,结果搜了一圈竟然找不到没找到存储自定义类型的办法。

好在在官方文档上找到解决方案:

@Entity
public class User { @Id private Long id; @Convert(converter = RoleConverter.class, dbType = Integer.class) private Role role; public enum Role { DEFAULT(0), AUTHOR(1), ADMIN(2); final int id; Role(int id) { this.id = id; } } public static class RoleConverter implements PropertyConverter<Role, Integer> { @Override public Role convertToEntityProperty(Integer databaseValue) { if (databaseValue == null) { return null; } for (Role role : Role.values()) { if (role.id == databaseValue) { return role; } } return Role.DEFAULT; } @Override public Integer convertToDatabaseValue(Role entityProperty) { return entityProperty == null ? null : entityProperty.id; } } } 

可以看到这个实体类里包含了一个自定义的枚举类型Role,在该类型上加了一个@Convert注解,括号里面指定了用于转换对象类型和数据库类型的converter类,以及该对象存储在数据库中的类型。

再来看看这个converter类做了什么事情。很简单,它实现了PropertyConverter接口,里面有两个方法,convertToEntityProperty是将数据库中的类型转换为java实体类;convertToDatabaseValue方法相反,将java实体类转换为数据库中的类型。我们只要在这两个方法里定义相应的转换规则即可。

看上去也不难,思路也很清晰。但是这个例子里的Enum类型要转换为int类型还是比较简单的,而笔者要存储的对象要复杂的多。这还是解决不了我的需求啊(欲哭无泪)。

如何快速简单的把一个复杂的数据类型转换为简单数据类型,而且还要能精准地转换回来?好像是有这么一个东西,json!!!json作为客户端和服务端之间数据传递的载体,确实满足我们现在的业务需求。更棒的是我们有gson这个解析框架来帮我们做转换!那么我们要做的事真就是无脑操作了。来看看我的Demo代码:

@Entity(
)
public class Zoo { indexes = { @Index(value = "zooId DESC, zoneId DESC", unique = true) } @Id(autoincrement = true) private Long id; @Property private Long zooId; @Property private Long zoneId; @Property @Convert(converter = CatConverter.class, columnType = String.class) private Cat cat; public static class CatConverter implements PropertyConverter<Cat, String> { @Override public Cat convertToEntityProperty(String databaseValue) { if (databaseValue == null) { return null; } return new Gson().fromJson(databaseValue, Cat.class); } @Override public String convertToDatabaseValue(Cat entityProperty) { if (entityProperty == null) { return null; } return new Gson().toJson(entityProperty); } } } 

这里我新建了一个叫Zoo的实体,里面包含一个Cat类型的对象,且不管该对象复杂与简单,我们甚至都不需要关心它的具体数据结构。加上@Convert注解后新建一个CatConverter类(注意converter类是内部类的话要声明为static),因为我们要转换为json存储起来所以数据库中的类型肯定是String了,标注好泛型,做一个参数的非空判断(良好习惯)。在转换的方法内部我们只需要利用gson将数据在java bean类型和json之间转换,就可以完成我们的需求了,是不是很简单呢?

Cat miaomiao = new Cat(13, "miaomiao");
Cat jingjing = new Cat(13, "jingjing"); ZooDao zooDao = daoSession.getZooDao(); zooDao.insertOrReplace(new Zoo(null, 222L, 333L, miaomiao)); zooDao.insertOrReplace(new Zoo(null, 222L, 331L, jingjing)); List<Zoo> zoos = zooDao.queryBuilder().list(); for (Zoo zoo : zoos) { Log.d("xxx", zoo.getZooId()+":"+zoo.getZoneId()+":"+zoo.getCat().toString()); } 

写完代码后make project自动生成新的ZooDao类(有兴趣的可以看看这个类,其实也挺简单的),不放心赶紧实验一下能不能直接存取自定义对象了。


作者:业松链接:https://www.jianshu.com/p/513fb2ba5485來源:简书著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

GreenDao存储自定义类型对象解决方案(转)相关推荐

  1. Java中Collection集合常用API - Collection存储自定义类型对象

    文章目录 Collection常用API Collection存储自定义类型对象 Collection常用API Collection集合API Collection是单列集合的祖宗接口,因此它的功能 ...

  2. HashSet存储自定义类型元素

    HashSet存储自定义类型元素 给HashSet中存放自定义类型元素时,需要重写对象中的hashCode和equals方法,建立自己的比较方式,才能保证HashSet集合中的对象唯一 创建自定义类 ...

  3. Qt基础之十:使用QVariant存储自定义类型

    QVariant支持大部分常见的Qt数据类型互转,比如: this->setProperty("number", 1); int number = this->prop ...

  4. 在NSUserDefaults中存储自定义类型的数据

    将自定义的类的数据以数组的形式直接存储到NSUserDefaults中会报错,需要进行转换,且需要将该类实现NSCoding协议. e.g. 存储过程 NSMutableArray *archiveA ...

  5. HashMap存储自定义类型键值: 重写HashCode和equals方法

    一个团体作为一个HashMap的key值,若团体成员的姓名年龄相同,则看作key值相同 因为是自定义类,所以需要重写HashCode和equals方法 public class RedVelvet { ...

  6. Redis开发:hash存储自定义Java对象及value的序列化器设置

  7. 在配置文件(.settings、.config)中存储自定义对象

    引言 我前面曾写过一篇<使用配置文件(.settings..config)存储应用程序配置>,我在其中指出"settings无法实现对一些复杂类型及自定义类型的支持". ...

  8. 游戏编程模式-类型对象

    "通过创建一个类来支持新类型的灵活创建,其每个实例都代表一个不同的对象类型." 动机 在RPG游戏中,我们通常会创建很多的怪物来作为我们主角的敌人,比如说恶龙.野狼等.怪物具有一系 ...

  9. SpringMVC-应用(数据绑定-自定义类型转换器,数据的格式化,数据校验)

    一.提出问题 SpringMVC封装自定义类型对象的时候,JavaBean要和页面提交的数据一一绑定.下面要知道: 1)页面提交的数据都是字符串 2)JavaBean中的属性如:Integer age ...

最新文章

  1. redistemplate分布式锁实现_基于 Redis SETNX 实现分布式锁
  2. 城市大脑的定义与理论基础综述
  3. 【转载】大连商品交易所-新套利撮合算法FAQ
  4. 第八周实践项目10 稀疏矩阵的十字链表表示
  5. Object.create()和new Object()
  6. 使用Boost.Compute的STL 在GPU上添加两个向量的实现
  7. 腾飞答不忘初心的三个问题
  8. 《JAVA 语言程序设计基础篇》chapter 5 方法
  9. java设计模式案例及使用
  10. 百度地图点聚合仿链家定位点多级聚合,且滑动、刷新加载定位点
  11. WinDynamicDesktop下载慢解决方法
  12. 计算机显示无法打开打印机添加,win10电脑提示无法打开添加打印机怎么办
  13. allegro修改铜皮网络
  14. 重磅开源!平安产险提出TableMASTER:表格识别大师
  15. 抖音 iOS 工程架构演进
  16. 交换a,b值的三种方法
  17. 计算机实训存在的问题和解决方法,中职学校计算机专业实训课程存在问题及的对策.doc...
  18. 【软件测试与质量保证】期末复习1(HITWH)(质量保证部分)
  19. matplotlib之pyplot模块之柱状图(bar():百分比堆积柱状图)
  20. 人脸识别算法源码SDK开发包人证比对二次开发检测核验开发包

热门文章

  1. OPENCV已知内参求外参
  2. 字符串排序 java_java字符串排序
  3. python面试总结 博客园_python面试题总结
  4. linux 负数_linux内核提权系列教程(2):任意地址读写到提权的4种方法
  5. 第55件事 产品疯传的7个基本原则
  6. 创建github账号
  7. Java 抽象类和接口
  8. AOP和IOC的作用
  9. 文件服务器vsftp的配置
  10. bash shell实现二进制与十进制数的互转