GreenDao存储自定义类型对象

需求场景: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类(有兴趣的可以看看这个类,其实也挺简单的),不放心赶紧实验一下能不能直接存取自定义对象了。

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,672评论 18 139
  • 最近公司项目选用GreenDao作为Android客户端本地数据库的对象关系映射框架。对于GreenDao虽然以往...
    业松阅读 10,918评论 17 18
  • (1) 当程玄把手链戴在我手上的那一刻,我有一种说不出来的感觉,怪怪的。 他说:“季凡,把手伸出来。” 他说:“季...
    猫左阅读 246评论 0 3
  • 敏感话题的搞笑版本 唐唐一年级的生活已经过了一半,转眼间第一个期末考试紧张的度过后,第一个寒假如期而至。小姑娘最近...
    Gabbi阅读 129评论 0 0
  • 我出生在一个以水稻种植为主的农村里,村子灌溉条件好,有溪有河。村里的人和睦勤劳,那时我的父母还算是个知识分子,都在...
    绘爱正面管教阅读 97评论 0 2