NEO

蜀道难,难于上青天!

导航

20130515-Grails In Action-3、建模(04小节)

Posted on 2013-05-16 00:59  页面载入出错  阅读(198)  评论(0编辑  收藏  举报

上一节建立了一个1:1的对象关系,这一节来建立一个1:m的对象关系

一个用户在一个论坛上或消息发布平台上可以发布多个消息,这个场景就是一个经典的1:m关系

1、建立一个Post对象

grails create-domain-class com.grailsinaction.Post

2、给Post对象增加属性

 1 package com.grailsinaction
 2 
 3 class Post {
 4     static belongsTo = [ user : User ]
 5     
 6     String content
 7     Date dateCreated
 8 
 9     static constraints = {
10         content(blank: false)
11     }
12 }

这里的级联关系和1:1是一样的,如果User对象的实体删除了,那么User实体对应的Post也将一并删除。这个动作是单向的,如果某个Post对象的实体删除了,User对象是不会删除的。

我们这里已经告诉Post对象和User对象的关系了,同样也要告诉User对象和Post对象的关系

3、修改User对象属性

1 package com.grailsinaction
2 
3 class User {
4     static hasMany = [ posts : Post ]
5     ......
6 }

加入这个关系后,告诉User,可以拥有多个Post,现在就可以用User.addToPosts()给User的实体增加Post实体,或者用User.removeFromPosts()删除Post实体,下面增加测试方法验证这个关系

4、增加一个Post集成测试类

grails create-integration-test com.grailsinaction.PostIntegration

5、增加一个测试方法

 1 package com.grailsinaction
 2 
 3 import static org.junit.Assert.*
 4 import org.junit.*
 5 
 6 class PostIntegrationTests {
 7 
 8     @Before
 9     void setUp() {
10         // Setup logic here
11     }
12 
13     @After
14     void tearDown() {
15         // Tear down logic here
16     }
17 
18     @Test
19     void testFirstPost() {
20         def user = new User(userId: 'joe', password: 'secret').save()
21         def post1 = new Post(content: "First post... W00t!")
22         user.addToPosts(post1)
23         
24         def post2 = new Post(content: "Second post...")
25         user.addToPosts(post2)
26         
27         def post3 = new Post(content: "Third post...")
28         user.addToPosts(post3)
29         
30         assertEquals 3, User.get(user.id).posts.size()
31     }
32 }

这里要注意,必须要调用User对象的save()方法,User对象才能将User持久化到数据库。一旦User对象保存了,并调用user.addToPosts()方法,Post对象会自动保持

6、运行命令测试结果

grails test-app -integration

当我们的帖子保存到数据库了,怎么能检索出来呢?

应该是先找到需要检索帖子的user对象,再将这个对象的所有post遍历出来,再建立一个集成测试看看结果

7、测试查询post实例

 1     @Test
 2     void testAccessingPosts() {
 3         def user = new User(userId: 'joe', password: 'secret').save()
 4         user.addToPosts(new Post(content: "First"))
 5         user.addToPosts(new Post(content: "Second"))
 6         user.addToPosts(new Post(content: "Third"))
 7         
 8         def foundUser = User.get(user.id)
 9         def postNames = foundUser.posts.collect { it.content }
10         assertEquals(['First', 'Second', 'Third'], postNames.sort())
11     }

新建了一个user对象joe,并给这个对象增加了3个post,使用User.get(id)方法获取到user对象,使用user.posts.collect{}闭包遍历user的post。对比时使用了sort()方法对遍历出来的posts排序。

这时会有个问题,一般显示post都会根据创建的时间降序排列,意思是最晚发布的帖子放在最前面,我们可以修改Post对象,让获取到post信息时就将序列排好,拿出来就是我们想要的结果

8、修改Post对象

1 ......
2     static mapping = {
3         sort dateCreated:"desc"
4     }
5 }

这样post返回的结果就是按照创建日期降序排列的了

假如我们是通过用户,再找某用户的帖子时,例如:user.posts.each,这样的场景就需要对用户下的帖子进行排序了,所以需要修改User对象,增加对用户下帖子的排序功能

1 ......
2     static mapping = {
3         profile lazy:false
4         posts sort:'dateCreated'
5     }
6 }

这种方式告诉我们,当我们根据用户去检索用户下的帖子信息时,按照这种方式排序