转自:http://www.jianshu.com/p/dbec25bd575f

前言

之前用过数据库框架: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。


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

android greendao3.0 多表关联关系讲解(转)的更多相关文章

  1. Android GreenDAO3.0——介绍

    引言 最近,学东西比较零散,各种知识混杂,于是记下学习记录,免得又忘了. 官方网址:http://greenrobot.org/greendao/documentation/introduction/ ...

  2. GreenDao3.0简单使用(转)

    关于GreenDao greenDao是一个将对象映射到SQLite数据库中的轻量且快速的ORM解决方案. 关于greenDAO的概念可以看官网greenDAO greenDAO 优势 1.一个精简的 ...

  3. GreenDao2.2升级GreenDao3.0的适配之路

    前言.为什么要升级到Greendao3.0? 1. 多人开发 以往的数据库建表建Dao等操作要新开一个module,在统一的地方管理数据库建表,现在可以直接写Entity.多人开发时自己管自己的Ent ...

  4. Android 6.0 运行时权限处理完全解析

    一.概述 随着Android 6.0发布以及普及,我们开发者所要应对的主要就是新版本SDK带来的一些变化,首先关注的就是权限机制的变化.对于6.0的几个主要的变化,查看查看官网的这篇文章http:// ...

  5. Google Android 6.0 权限完全解析

    注:本文只针对Google原生Android系统有效, 小米魅族等手机有自己的权限机制, 可能不适用 一.运行时权限的变化及特点 新的权限机制更好的保护了用户的隐私,Google将权限分为两类,一类是 ...

  6. Android 5.0 Default SMS App以及运营商授权SMS App

    已同步更新至个人blog:http://dxjia.cn/2015/08/android-5-default-sms-app/ 题外话:博友们有没有好用的写博客客户端推荐啊,cnblogs推荐的win ...

  7. Android 4.0 事件输入(Event Input)系统

    参考:http://blog.csdn.net/myarrow/article/details/7091061 1. TouchScreen功能在Android4.0下不工作 原来在Android2. ...

  8. myBatis 基础测试 表关联关系配置 集合 测试

    myBatis 基础测试 表关联关系配置 集合 测试 测试myelipse项目源码 sql 下载 http://download.csdn.net/detail/liangrui1988/599388 ...

  9. android 7.0 多渠道打包 - 美团开源工具Walle 命令行打包

    在Android 7.0(Nougat)推出了新的应用签名方案APK Signature Scheme v2后,之前快速生成渠道包的方式(美团Android自动化之旅-生成渠道包)已经行不通了,对此美 ...

随机推荐

  1. yum安装openresty

    在群里看到春哥发的,先记录下来.一切都以官网为准,以后安装部署生态会越来越完善的. OpenResty 官方现在开始维护自己的打包虚机集合了,新的 linux 包仓库正在陆续登陆 openresty. ...

  2. PHP缩略图类

    class ThumbImages{ /** * 生成缩略图 * prorate 按比例缩放 * distortion 扭曲型缩图 * cut 最小裁剪后的缩图 * backFill 背景填充图 * ...

  3. steelray project viewer

    steelray project viewer是一款英文语言软件,透过Steelray Project Viewer,可以打开.导航.浏览.打印Microsoft Project的.mpp文件.

  4. Windows下将ISO镜像制作成U盘启动的工具(U盘启动工具/UltraISO/Rufus/Universal-USB)

    说明:基于Windows的U盘启动制作都是非常的简单,在软件上指定ISO文件之后,一般都是选择写入到哪个U盘即可. 1.UltraISO 2.Rufus 3.Universal-USB 4.大白菜

  5. SQL 存储过程入门(事务)

    本篇我们来讲一下事务处理技术. 为什么要使用事务呢,事务有什么用呢,举个例子. 假设我们现在有个业务,当做成功某件事情的时候要向2张表中插入数据,A表,B表,我们插入的顺序是先插入A,再插入B表,如果 ...

  6. 完全理解Gson(3):Gson反序列化

    完全理解Gson(2):Gson序列化 完全理解Gson(1):简单入门 本文延续前一篇文章,继续介绍简单基本的Gson用法.这篇文章我们将介绍如何将复杂的JSON对象解析为Java对象,其中Java ...

  7. UVA 1665 Islands

    题意:输入一个n*m矩阵,每一个格子都有一个正整数,再输入T个整数ti,对于每一个ti,输出大于ti的正整数组成多少个四连快 思路:正着做的话事实上相当于删除连通块,而假设反着做的话就相当于变成添加连 ...

  8. oracle find blocking session

    show current session id     select sid from v$mystat where rownum=1; show blocking session     selec ...

  9. 从头认识Spring-1.7 如何通过属性注入Bean?(1)-如何通过属性向对象注入值?

    这一章节我们来讨论一下如何通过属性注入Bean? 这一章节分为两部分,第一部分我们通过属性向对象注入值,第二部分我们通过属性向对象注入还有一个对象的引用. 1.如何通过属性向对象注入值? (1)dom ...

  10. 改进xutils下载管理器,使其,在随意地方进行进度更新,以及其它状态监听操作

    1.前面在做下载进度监听.尝试过,通过加入 弱引用的View进度条,到相应的集合. 等到要进行更新进度的时候.通过Key 获取相应的VIew来进行更新 进度条.效果是达到了,可是我们怎样来监听其它的状 ...