Groovy&Grails-代码剪辑-对象关系

One To One

一张脸上只有一个鼻子。


class Face {
    Nose nose
}

class Nose {
}

这是一个简单的单向关联,从face对象可以查到这个nose的信息,从nose就没办法了。如果要达到这个效果,需要把这个关联修改成双向的,代码如下:


class Face {
    Nose nose
}

class Nose {
    Face face
}

这样双向查询就没有问题了,但是不能级联更新,如果需要更新需要一个对象一个对象的增加。如果我们在实例化Face的时候同时把Nose也增加一条信息,则需要增加一个级联更新属性,代码如下:


class Face {
    Nose nose
}

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

在这里,我们使用belongsTo 表示Nose “属于” Face。其结果就是我们创建并保存它时,数据库可以_级联_更新/插入 Nose,实例化代码如下:


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

上面这段代码会将fase和nose都保存到数据库,需要注意的是反向操作是不行的,错误代码如下:


new Nose(face:new Face()).save() // 这样会报错

belongsTo 的另一个重要作用是,如果你删除了一个Face 实例,那么相关的Nose也会被删除,操作代码如下:


def f = Face.get(1)
f.delete() // Face和Nose实例都会被删除

如果没有 belongsTo 的话,将不会级联删除,你会得到一个外键约束的错误,除非你手工去删除Nose。


// 如果没有belongsTo会报错,因为有约束
def f = Face.get(1)
f.delete()

// 所以没有belongsTo时需要手工删除
def f = Face.get(1)
f.nose.delete()
f.delete()

如果在删除Face实例的时候,我们不想自动删除和face关联的nose实例,需要将belongsTo修改一下,代码如下:


class Face {
    Nose nose
}

class Nose {
    static belongsTo = Face
}

注意:我们没有用明确的语法声明映射关系,所以这个关联是单向的,并且不能设置成双向,否则会出现映射错误。

One to Many

一对多的关系是指,当一个类(比如Author)拥有另一个类(Book)的多个实例这种情况。在Grails中,你可以使用 hasMany 来定义这种关联


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

class Book {
   String title
}

这样我们有了一个一对多的单向关联。Grails在数据库级别将默认使用外键映射来映射这种关联关系。Grails 将会根据hasMany 设置为domain类自动注入一个类型为java.util.Set的属性。这个可以被用来遍历集合,代码如下:


def a = Author.get(1)
a.books.each {
    println it.title
}

上面的代码遍历所有a作家的所著的所有图书。在上面的关联关系下,book对象可以级联保存和更新,但是不能级联删除,如果删除author的某个实例还要将和这个实例相关的book也删除,则需要使用belongTo关键字,代码如下:


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

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

其数据操作方法就可以参考OneToOne了。

在数据映射对象中,如果对象有多个属性有映射关系,则需要使用mappedBy关键字来指定他们分别映射的是哪个集合,代码如下:


class Airport {
    static hasMany = [flights:Flight] //一个机场有多条航线
    static mappedBy = [flights:"departureAirport"] //离港机场
}
class Flight { //一条航线涉及到入港机场和出港机场
    Airport departureAirport
    Airport destinationAirport
}

一条航线涉及到入港机场和出港机场,针对哪条航线的入港机场是什么,出港机场是什么,其映射关系需要指定。

更为明细的指定如下:


class Airport {
    static hasMany = [outboundFlights:Flight, inboundFlights:Flight] //入站航班和出站航班
    static mappedBy = [outboundFlights:"departureAirport", inboundFlights:"destinationAirport"]
}
class Flight {
    Airport departureAirport
    Airport destinationAirport
}

Many to Many

这种关联需要在关联的两方都定义hasMany,并在关联的被拥有方定义belongsTo(可用来指某物属于某人):


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

Grials在数据库层使用连接表来映射多对多关联。关联的拥有方,在这里是Author,负责持久化这个关联,并且它是唯一可以级联保存对方的一方。

比如下面的代码可以工作,并且会级联保存:


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

但是下面的代码只保存 Book 而不保存authors!


new Book(name:"Groovy in Action")
    .addToAuthors(new Author(name:"Dierk Koenig"))
    .addToAuthors(new Author(name:"Guillaume Laforge"))     
    .save()

这正是我们期望的行为,跟Hibernate中一样,多对多关联中只有一方可以管理关联。

你可能感兴趣的:(Groovy&Grails-代码剪辑-对象关系)