Grails MongoDB插件如何处理多态?

时间:2013-06-13 17:12:00

标签: mongodb grails gorm polymorphic-associations

我有一个如下对象:

class User {
    static mapWith="mongo"
    static embedded = [ 'profiles' ]

    String email
    List<Profile> profiles;
}

interface Profile {
}

class Profile1 implements Profile {
}

class Profile2 implements Profile {
}

如果我将具体类Profile1或Profile2添加到User对象并将其保存到数据库,则会在从MongoDB中读取该对象时抛出异常。我没有看到任何信息被保存到数据库中,以确定在这种情况下它应该实例化的对象类型。并且有关于如何处理此案例的ZERO文档。其他框架有处理这个的机制,因此Grails MongoDB被严重破坏,或者这只是未记录的(再次)。那么我该如何解决这个问题?

例外情况如下:

| Error 2013-06-12 18:48:00,390 [http-bio-8080-exec-5] ERROR errors.GrailsExceptionResolver  - InstantiationException occurred when processing request: [POST] /mpa/user/authenticate -parameters:

  email: carhubb@gmail.com
  password: ***
  com.mycompany.security.Profile. Stacktrace follows:
  Message: com.mycompany.security.Profile
  Line | Method
  ->>  342 | newInstance0                        in java.lang.Class
  - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 
  |    310 | newInstance                         in     ''

2 个答案:

答案 0 :(得分:1)

Grails MongoDB完全没有破坏,每个用例都可能没有记录。 :)
您的用例按预期正常工作并成功测试,如下所示。

// src/groovy
interface Profile {
    Integer getDuration()
}

import org.bson.types.ObjectId
class Profile1 implements Profile {
    ObjectId id
    String profileName
    String type
    Date effectiveStartDate
    Date effectiveEndDate

    Integer getDuration(){
        effectiveEndDate - effectiveStartDate
    }

    static mapWith = "mongo"
}

import org.bson.types.ObjectId
class Profile2 implements Profile{
    ObjectId id
    String profileName
    String type
    Date effectiveStartDate
    Date effectiveEndDate

    static mapWith = "mongo"

    Integer getDuration(){
        effectiveEndDate - effectiveStartDate
    }
}
class User {
    ObjectId id

    static mapWith = "mongo"
    static embedded = ['profiles']

    String email
    List<Profile> profiles
}

class UserController {

    def index() {
        def profile1 = new Profile1(type: 'Individual',
                                    profileName: 'IndividualProfile',
                                    effectiveStartDate: new Date(),
                                    effectiveEndDate: new Date() + 100) as Profile
        def profile2 = new Profile2(type: 'Company',
                                    profileName: 'CompanyProfile',
                                    effectiveStartDate: new Date(),
                                    effectiveEndDate: new Date() + 50) as Profile

        println profile1.duration //prints 100
        println profile2.duration //prints 50

        def user = new User(profiles: [profile1, profile2], email: 'abc@gmail.com').save(flush: true)

        render user as JSON
    }
}

//db.user.find()
{ 
    "_id" : ObjectId("51ba8d55892cb98368b2f1e5"), 
    "email" : "abc@gmail.com", 
    "profiles" : [{   
            "effectiveEndDate" : ISODate("2013-09-22T03:26:13.396Z"),   
            "effectiveStartDate" : ISODate("2013-06-14T03:26:13.396Z"),   
            "profileName" : "IndividualProfile",    
            "type" : "Individual" 
        },
        {   
            "effectiveEndDate" : ISODate("2013-08-03T03:26:13.423Z"),       
            "effectiveStartDate" : ISODate("2013-06-14T03:26:13.423Z"),   
            "profileName" : "CompanyProfile",
            "type" : "Company" 
        } 
    ], 
    "version" : 0 
}

您还可以找到上述设置here

注意: - 为简化使用,Profile1Profile2设计相似。

答案 1 :(得分:1)

实际的答案是grails似乎处理类而不是接口,这真的很奇怪,因为如果你处理类的多态性,那么为接口处理它是微不足道的,因为你可以用同样的方式处理它。但是如果你为所有引用类型使用类,它将为mongodb添加一个特殊的'_class'属性,它将使用它来实例化对象在保存时指向的实际引用的对象。现在,在一个段落中解释而不是通过源代码和单元测试页面进行解释有多难?