本文来自:fair-jm.iteye.com 转截请注明出处
来自:
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]
}
这样在数据库中建立的表是:
可见外键是由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
我把这个问题详细说一下 按以上代码输入后:
表结构:
执行下列代码:
import orm_test.*
Person.withTransaction{
new Person(name:"cc",parent:new Person(name:"fairjm")).save()
}
同一个关系的不同方向 就如上图所示
用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默认用一张中间表的形式:
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
}
以上设置的话 表的结构会发生变化 不会有中间表了 外键字段被加在了多的一端:
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
}
多对多生产的中间表:
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()
}
文档里还说 多对多关系现在的脚手架还不支持
以后的内容有空再更新了~~