可以使用Grails派生属性来检索域对象吗?

时间:2016-12-06 23:41:19

标签: grails gorm

我有Request的域类,RequestSet(由单个用户同时提交的各个请求的集合)和请求Status。我想聚合请求的状态,以便请求集的状态由其各个请求的状态确定。

这些域类的相关部分如下所示:

class RequestSet {
    Long id
    static hasMany = [requests: Request]
}

class Request {
    Long id
    Status status
}

class Status {
    Long id
    String status
}

我尝试将派生的status属性添加到RequestSet类,如下所示:

class RequestSet {
    Long id
    Status status
    static hasMany = [requests: Request]

    static transients = ['status']

    static mapping = {
        status formula: '(SELECT MIN(status_id) FROM (SELECT status_id FROM request rq WHERE rq.request_set_id = id))'
    }
}

但是,当我运行我的应用程序并检查requestSet对象的属性时,状态都为空。

我是否正在尝试使用派生属性来实现?我可以使用getter解决一个瞬态属性,但我希望能够使用动态查找器根据它们的冒泡状态检索requestSets。我见过的所有例子都使用了公式的结果;我想知道我是否能以某种方式让Grails将返回的值解释为Status类的ID。

编辑,2016年12月7日:请注意,我将派生的status属性标记为瞬态。如果不这样做,我会从Hibernate获得ClassCastException

org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'transactionManagerPostProcessor': Initialization of bean failed; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'transactionManager': Cannot resolve reference to bean 'transactionManager_companion' while setting constructor argument with key [1]; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'transactionManager_companion': Cannot resolve reference to bean 'sessionFactory_companion' while setting bean property 'sessionFactory'; nested exception is org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'sessionFactory_companion': Invocation of init method failed; nested exception is java.lang.ClassCastException: org.hibernate.mapping.Formula cannot be cast to org.hibernate.mapping.Column
    at java.util.concurrent.FutureTask.run(FutureTask.java:266)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
    at java.lang.Thread.run(Thread.java:745)

我怀疑即使公式有效,我也无法使用具有此属性的动态查找器,因为它被标记为瞬态。我会满足于使用公式计算状态的ID,但我仍然遇到问题。

现在我的域类看起来像这样:

class RequestSet {
    Long id
    Long statusId
    static hasMany = [requests: Request]

    static mapping = {
        statusId formula: '(SELECT MIN(status_id) FROM (SELECT rq.status_id FROM request rq WHERE rq.request_set_id = id))'
    }
}

获取请求集时公式生成的SQL是

(SELECT MIN(this_.status_id) FROM (SELECT rq.status_id FROM request rq WHERE rq.request_set_id = this_.id)) as formula3_0_ from request_set

我得到SQLSyntaxErrorExceptionORA-00904: "THIS_"."STATUS_ID": invalid identifier。如何重写公式以便“this_”。没有添加到status_id?

1 个答案:

答案 0 :(得分:0)

在回答我的原始问题时,不,您不能使用公式来派生属性是另一个域类。我无数次失败的尝试使我得出了这个结论,this blog的作者似乎也这么认为。具体来说,“派生属性可以是Hibernate支持的任何类型”而我的Status类不是。

至于我的后续问题,根据this answer for a different question,HQL不支持我使用的公式。我设法使用此公式将statusId作为派生属性:

(SELECT MIN(status.id) FROM status WHERE status.id IN (SELECT rq.status_id FROM request rq WHERE rq.request_set_id = id))

感谢@Mike提出打开SQL日志记录的建议。这确实有助于弄清楚发生了什么。