android greendao3.0 多表關聯關係講解

來源:互聯網
上載者:User

標籤:關於   entity   tac   lib   獲得   tom   com   之間   商業   

轉自: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),就報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;
//中間表   “第三者”@Entitypublic class JoinStudentToPerson {   @Id(autoincrement = true)    private Long id;    //和person關聯的id    private Long pid;    //和student關聯的id    private Long sid;}
@Entitypublic 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 多表關聯關係講解(轉)

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.