Android greendao3.0 多表关联踩坑实践

前言

之前用过数据库框架:realm、kjdb,今天准备实践学习一下greendao 3.0。
greendao 3.0之前的版本有很大的不同,主要是增加了annotation注解,然后表之间和对象之间的关系也通过注解而变得更加灵活方便了。以前用过旧版本的都知道,对于多表多对象之间的关联,要写的代码不少。

我在学习greendao 3.0的时候,有一个感触,网上的文章很多,但是千篇一律,大多都是翻译官方文档而来,举得例子可谓是“无一例外”。网上搜罗了半天,收藏几篇比较好的帖子。
史上最高效的ORM方案——GreenDao3.0详解
Android ORM——初识greenDAO 3及使用greenDAO 3前应该掌握的一些知识点(一)

特别是关于3.0对象多表多对象关联的博客更是没有,所以打算自己实践学习然后总结分享一番。

踩坑

主要踩了两个坑:

  • greendao的关联关系是通过主外键(对象之间关联的id)来构建的。realm是直接通过对象关系来自动构建的。
  • 如果属性是List<> xx, greendao不会自动调用设置xx的值,只有手动调用getXX的时候获取.我在打印log的时候被坑惨了,无论怎么样都是为null.

发现

  • 首先bean类,会自动生成一些方法,比如get set 构造方法 getSession等
  • 如果是list或者数组类型的属性XX,只有getXX方法,没有setXX方法
  • 如果你要打印bean类的toString方法,这里要调用getXXX方法,而不是直接打印对象(因为没有赋值,而是在getXX的时候才赋值的)
    @Override
    public String toString() {
        return "Person{" +
               "students=" + getStudents() +  //这里不是直接students
                ", id=" + id +
                ", cardId=" + cardId +
                ", age=" + age +
                ", name='" + name + '\'' +
                ", average=" + average +
                ", cid=" + cid +
                ", cls=" + cls +
                ", fid=" + fid +
                ", friends=" + getFriends() +
                '}';
    }
  • greendao支持 自身对自身的关系关联,比如

正文: 多表映射关联

一对一:比如一个人有一个头

如果是按照以往的对象关系数据库

#person类中

   @Id(autoincrement = true)
    private Long id;
    private String name;

//    private Long hid;
    @ToOne
    private Head head;
  Head head = new Head();
        head.setId(13l);
        head.setName("head");

        Person p = new Person();
        p.setId(null);
        p.setHead(head);//直接设置对象
        p.setName("jafir");

直接写对象,然后setHead(head) 就搞定了。但是在greendao中一切对象关联关系都是通过主外键来实现的。应该改为如下:

#person类中
  @Id(autoincrement = true)
    private Long id;
    private String name;

    private Long hid;//这是与头关联的外键
    @ToOne(joinProperty = "hid") //这个是注解绑定 hid就是上面一行的hid
    private Head head;//对象,但是不需要setHead

build之后,就有setHid的方法了,我们的对象关联不采用setHead,而是setHid

PersonDao personDao = GreendaoHelper.getDaoSession().getPersonDao();
        HeadDao headDao = GreendaoHelper.getDaoSession().getHeadDao();

        Head head = new Head();
        head.setId(13l);//这里的head id和person里的hid一样
        head.setName("head");

        Person p = new Person();
        p.setId(null);
        p.setHid(13l);//这里的hid是head的id,就是这样通过id构建起关联的
        p.setName("jafir");

        headDao.insert(head);
        personDao.insert(p);

        List<Person> persons = personDao.queryBuilder().build().list();
        for (Person person : persons) {
            Log.d("debug","person:"+person.toString());
        }

注意:person的toString()方法系统生成的需要修改一下:

 @Override
    public String toString() {
        return "Person{" +
                "head=" + getHead() +//这里需要改为getHead
                ", name='" + name + '\'' +
                ", id=" + id +
                '}';
    }
tips:
  • bean的id最好用Long类型而不是long
 @Id(autoincrement = true)
private Long id;
Person p = new Person();
        p.setId(null);
        p.setHid(14l);
        p.setName("jafir");

因为,如果Long,我们设置了id是自增长,我们可以setId(null),便是自增长。如果是long类型,你设置setId(null),就报空指针。

一对多:比如一个老师有多个学生

#teacher类中
 @ToMany(referencedJoinProperty = "tid")//指定与之关联的其他类的id
 private List<Student> studnets;
#student类中
  @Id
  private Long id;
  private Long tid;//这个就是外键 就是person的id

关系描述:
多个学生都有同一个老师,所以每个学生的tid,就应该是同样的,并且tid 就是老师的id ,这样就构成了1对多的关系

注意:学生的自增长id ,跟与老师关联的tid是不一样的,两码事

一对多:比如一个人有一群朋友(朋友也是person)

按照我们上面的思路,那么person类里面就应该有一个外键指向自身的主键id

#person类中
 private Long id;//自身id
 private Long fid;//外键关联id
 @ToMany(referencedJoinProperty ="fid" )//指定与之关联的其他类的id
 private List<Person> friends;

如果一个人的id是1,他有3个朋友,那么friends里面person的fid都是1,这样这个人调用getFriends就能拿到自己的3个朋友。主要就是通过设置id来构建关联关系的。

多对多:

多对多的话就比较复杂,不是两个表或者两个对象直接关联,而是要通过一个“第三者”

#person类中
    @Id(autoincrement = true)
    private Long id;
    private String name;
   // 对多,@JoinEntity注解:entity 中间表;sourceProperty 实体属性;targetProperty 外链实体属性
    @ToMany
    @JoinEntity(
            entity = JoinStudentToPerson.class,
            sourceProperty = "pid",
            targetProperty = "sid"
    )
    private List<Student> students;
//中间表   “第三者”
@Entity
public class JoinStudentToPerson {
   @Id(autoincrement = true)
    private Long id;
    //和person关联的id
    private Long pid;
    //和student关联的id
    private Long sid;
}
@Entity
public class Student {
    @Id
    private Long id;
    private String name;
    // 对多,@JoinEntity注解:entity 中间表;sourceProperty 实体属性;targetProperty 外链实体属性
    @ToMany
    @JoinEntity(
            entity = JoinStudentToPerson.class,
            sourceProperty = "sid",
            targetProperty = "pid"
    )
    private List<Person> persons;
}

然后测试代码

  private void test() {

        PersonDao personDao = GreendaoHelper.getDaoSession().getPersonDao();
        HeadDao headDao = GreendaoHelper.getDaoSession().getHeadDao();
        StudentDao studentDao = GreendaoHelper.getDaoSession().getStudentDao();
        JoinStudentToPersonDao spDao = GreendaoHelper.getDaoSession().getJoinStudentToPersonDao();

//        Head head = new Head();
//        head.setId(14l);
//        head.setName("head");

        Person p1 = new Person();
        p1.setId(1l);
        p1.setName("jafir1");
        Person p2 = new Person();
        p2.setId(2l);
        p2.setName("jafir2");
        Person p3 = new Person();
        p3.setId(3l);
        p3.setName("jafir3");


        Student stu1 = new Student();
        stu1.setId(1l);
        stu1.setName("stu1");
        Student stu2 = new Student();
        stu2.setId(2l);
        stu2.setName("stu2");
        Student stu3 = new Student();
        stu3.setId(3l);
        stu3.setName("stu3");


        // 模拟 多对多关系
        // 假如 p1有3个:stu1\stu2\stu3
        //  stu1 stu2 stu3 都有2个 :p1\p2

        //p1有stu1 stu2 stu3     那么反过来stu123都有p1
        JoinStudentToPerson sp1 = new JoinStudentToPerson();
        sp1.setPid(1l);
        sp1.setSid(1l);
        JoinStudentToPerson sp2 = new JoinStudentToPerson();
        sp2.setPid(1l);
        sp2.setSid(2l);
        JoinStudentToPerson sp3 = new JoinStudentToPerson();
        sp3.setPid(1l);
        sp3.setSid(3l);

        //p2有stu1 stu2 stu3     那么反过来stu123都有p2
        JoinStudentToPerson sp4 = new JoinStudentToPerson();
        sp4.setPid(2l);
        sp4.setSid(1l);
        JoinStudentToPerson sp5 = new JoinStudentToPerson();
        sp5.setPid(2l);
        sp5.setSid(2l);
        JoinStudentToPerson sp6 = new JoinStudentToPerson();
        sp6.setPid(2l);
        sp6.setSid(3l);

        spDao.insert(sp1);
        spDao.insert(sp2);
        spDao.insert(sp3);
        spDao.insert(sp4);
        spDao.insert(sp5);
        spDao.insert(sp6);

        personDao.insert(p1);
        personDao.insert(p2);
        personDao.insert(p3);

        studentDao.insert(stu1);
        studentDao.insert(stu2);
        studentDao.insert(stu3);

//        headDao.insert(head);
//        personDao.insert(p1);

        List<Person> persons = personDao.queryBuilder().build().list();

        for (Person person : persons) {
            Log.d("debug","person:"+person.toString());
        }
    }

注意:在person和student的toString里面不能都写getStudents getPerson,不然会你调我,我调你,然后死循环,出现log打印Stack Overflow。只能单独打印测试结果

//打印person的测试结果
Person{
students=[
Student{name='stu1', id=1}, 
Student{name='stu2', id=2},
 Student{name='stu3', id=3}], 
head=null, hid=null, name='jafir1', id=1}

Person{
students=[
Student{name='stu1', id=1}, 
Student{name='stu2', id=2}, 
Student{name='stu3', id=3}],
 head=null, hid=null, name='jafir2', id=2}

Person{
students=[], 
head=null, hid=null, name='jafir3', id=3}

students的测试结果就不列出了。
总之实现起来就是这样。

后记

在探索3.0版本的时候,确实碰了不少壁,踩了很多坑,遇到了很多疑惑,但是通过自己不断地摸索探究测试,最终还是找到了解决的方法。希望对大家有用,于是分享出来,欢迎大家指正。

这个还要再说一下自己的看法:
对于greendao或者realm,个人觉得,realm确实更高级一点,是直接对象关联的,用起来也更方便一点,而且有很好的中文文档。greendao呢,其实还算是比较原始的数据库框架,但是它最大的优点就是效率高

所以,如果你的数据量大要求效率,你应该使用greendao,不然简小的数据库还是建议使用realm。

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

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,587评论 18 399
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,527评论 25 707
  • 世间单身千百种,最苦逼、最活该非暗恋莫属,在对方毫不知情的情况下,一个人自导自演,难受的死去活来,一出独角戏跳得比...
    Prudence19阅读 169评论 0 0
  • 1月23日,经排查发现: 一,王楼村万家超市 问题:无登记台账, 措施:立即整改 责任人:王明岳 二,小许夫广烟酒...
    谢炉镇食安办阅读 305评论 0 0
  • 昨天,有会员找我倾诉孩子的问题。她觉得儿子有拖延的习惯,不爱完成作业。他经常被老师留下来,因为没有完成作业,他也是...
    CrystalatBJ阅读 281评论 0 3