Grails GORM中的映射(一对一 一对多 多对多)

purpen 2013-11-25

来自:

http://grails.org/doc/latest/guide/single.html#GORM

配合手册里的例子 做了些小试验 

这篇博文是里面6.1~6.2的内容 官方的指南写得很详细很好啊...

Grails的ORM的底层实现还是用的hibernate(在现在最新的2.3.3中是hibernate3)

快速开始:

grails create-domain-class helloworld.Person

增加属性:

class Person {
    String name
    Integer age
    Date lastVisit
} //id是自动生成的 所以不用自己创建(当然自己设定个 Integer id 是完全没问题的)

基本的CRUD操作:

创建(c):

def p = new Person(name: "Fred", age: 40, lastVisit: new Date())
p.save()

读取(R):

def p=Person.get(1) //1是id
assert 1==p.id

用只读方式读取

def p=Person.read(1)

用代理(proxy)方式读取(这和hibernate中的load是一样的):

def p = Person.load(1)

更新(U):

U的操作就是先读再存:

def p = Person.get(1)
p.name = "Bob"
p.save()

删除(D):

def p = Person.get(1)
p.delete()

6.2中是关于一对一 一对多 多对多的内容:

多对一和一对一关系:

//Face类
class Face {
    Nose nose
}

//nose类
class Nose {
    static belongsTo = [face:Face]
}

这样在数据库中建立的表是:

 
Grails GORM中的映射(一对一 一对多 多对多)
Grails GORM中的映射(一对一 一对多 多对多)
 可见外键是由face维护的
 

保存的话通过:

new Face(nose:new Nose()).save()

因为belongsTo是将控制权给对方 所以

new Nose(face:new Face()).save() // will cause an error

是错误的 face还是瞬时态的 不能被保存

删除Face的话对应的Nose也会被删除

def f = Face.get(1)
f.delete() // both Face and Nose deleted

但如果让一个Nose被多于一个的Face关联

那么删除任意一个Face会抛出异常的(那个Nose还被其他的Face关联)

所以这样的做法不是真正的一对一 而是nose对face的一个一对多关系 为了保险 限制一下nose的唯一

static constraints={
 nose(unique:true)
}

Face对Nose是级联的保存 更新和删除

真正的一对一:

class Face {
    static hasOne = [nose:Nose]
}

class Nose {
    Face face
}

这样的话 外键字段是在nose中的face_id中

在Nose中是不能级联保存Face的

有了hasOne就不能单独保存任何一方了

在试验中 不设置Face的Nose属性 不设置Nose的Face属性都不能被正常的save

删除face nose也会被级联删除

不过单独删除Nose是没问题的

Nose.withTransaction{
 Nose.get(2).delete() //删除一个Nose
}

是可行的 从face中查询也只是得到null而已

自引用的例子:

如果有多个相同类型的属性在里面 并且这些属性都是自引用的(引用自身表)

那么grails的映射关系就需要人为指定一下:

class Person {
    String name
    Person parent
    static belongsTo = [ supervisor: Person ]
    static constraints = { supervisor nullable: true }
}

他会把parent和supervisor当作是同一个联系的两个不同的方向

如果你在A1上设置了parent A2 那么会在A2上设置supervisor为A1

我把这个问题详细说一下 按以上代码输入后:

表结构:


Grails GORM中的映射(一对一 一对多 多对多)
 

执行下列代码:

import orm_test.*

Person.withTransaction{
 new Person(name:"cc",parent:new Person(name:"fairjm")).save()
}

   Grails GORM中的映射(一对一 一对多 多对多)

  同一个关系的不同方向 就如上图所示

 
 

用mappedBy可以解决:

class Person {
    String name
    Person parent
    static belongsTo = [ supervisor: Person ]
    static mappedBy = [ supervisor: "none", parent: "none" ]
    static constraints = { supervisor nullable: true }
}

mappedBy是指导如何映射的 这样的"none"是表示不映射对方类(也就是Person啦)的任意一个属性

(最开始 grails会将Parent映射到对方的supervisor的

也就是类似:

static mappedBy = [ supervisor: "parent", parent: "supervisor" ]

)

关于一个类中有多个相同类型的属性的映射问题都要通过mappedBy来手工指定映射关系解决掉

一对多:

class Author {
    static hasMany = [books: Book]
    String name
}
class Book {
    String title
}

这是一个单向的一对多 grails默认用一张中间表的形式:


Grails GORM中的映射(一对一 一对多 多对多)
 

def a = Author.get(1)
for (book in a.books) {
    println book.title
}

默认是懒加载的形式

这样做的默认会级联更新和保存 但不会级联删除

import orm_test.*

Author.withTransaction{
/* def au=new Author(name:"cc")
 au.books=[new Book(title:"book1"),new Book(title:"book2")]
 au.save()*/
 Author.get(1).delete() //不会导致book被删除
}

实现级联删除要用belongsTo 原先的单向关系也会变成一个双向的一对多关系:

class Author {
    static hasMany = [books: Book]
    String name
}
class Book {
    static belongsTo = [author: Author]
    String title
}

 以上设置的话 表的结构会发生变化 不会有中间表了 外键字段被加在了多的一端:

  Grails GORM中的映射(一对一 一对多 多对多)
  

import orm_test.*

Author.withTransaction{
/*
 def au=new Author(name:"cc")
 au.books=[new Book(title:"book1",author:au),new Book(title:"book2",author:au)]  //这边必须显示指定au了 book那多了一个属性不是~~ 不过没有addToBooks方法 但多对多的时候就会有..
 au.save()
 */
 Author.get(7).delete() //会导致book被删除
}

用mappedBy解决一个类下多个同类型属性的问题:

class Airport {
    static hasMany = [outboundFlights: Flight, inboundFlights: Flight]
    static mappedBy = [outboundFlights: "departureAirport",
                       inboundFlights: "destinationAirport"]
}
class Flight {
    Airport departureAirport
    Airport destinationAirport
}

多对多:

class Book {
    static belongsTo = Author
    static hasMany = [authors:Author]
    String title
}


class Author {
    static hasMany = [books:Book]
    String name
}

 多对多生产的中间表:


Grails GORM中的映射(一对一 一对多 多对多)

Author负责关系的维护 

(Book是不能向author表插入新Author的 也就是Book不能在addToAuthors的时候用一个瞬时态的对象)

new Author(name:"Stephen King")
        .addToBooks(new Book(title:"The Stand"))
        .addToBooks(new Book(title:"The Shining"))
        .save()

 这样是可以的

book不会被级联删除..

实验了下

实验一:

Author.withTransaction{
def p=new Author(name:"cc")
p.addToBooks(new Book(title:"book1")).addToBooks(new Book(title:"book2")).save() //可以级联保存book
}
Book.list()
Author.withTransaction{ //用户删除的话 中间表的条目会被删除 但是book不会被删除
for( def a:Author.list())
{
 a.delete(); //删除所有用户
}
}
Book.list() //还存在

实验二:

import orm_test.*
Author.withTransaction{
 def p=new Author(name:"cc")
 p.addToBooks(new Book(title:"book1")).addToBooks(new Book(title:"book2")).save()
}
Author.withTransaction{
 def p=new Author(name:"cc2") //新增一个用户cc2
 p.save()
}
Book.withTransaction{
Book.findByTitle("book1").addToAuthors(Author.findByName("cc2")).save() //中间表更新了状态
}

查询也是没有问题的:

import orm_test.*
Book.withTransaction{
def b=Book.findByTitle("book1")
b.authors?.toString()
}

 文档里还说 多对多关系现在的脚手架还不支持

  

以后的内容有空再更新了~~

相关推荐