Grails 域模型中的继承

Dón*_*nal 6 grails grails-orm

我的 Grails 应用程序的域模型具有以下要求:

  • 一个用户属于零个或一个组织
  • 组织是慈善机构或公司
  • 慈善机构和公司有一些共同的字段,还有一些(不可为空的)字段,这些字段对于每个组织类型都是独一无二的

我将常见的组织字段放入一个抽象Organisation类中,Charity并且Company两者都进行了扩展。我无法将此层次结构存储在单个表中,因为存在特定于每个组织类型的不可为空的字段。域模型的相关部分如下所示:

class User {
  String name

  static belongsTo = [organization: Organization]

  static constraints = {
    organization nullable: true
  }
}

abstract class Organization {    
    String name

    static hasMany = [users: User]

    static mapping = {
        tablePerHierarchy false
    }
}

class Charity extends Organization {
  // charity-specific fields go here
} 

class Company extends Organization {
  // company-specific fields go here
}
Run Code Online (Sandbox Code Playgroud)

看这个模型生成的MySQL schema,组织-公司和组织-慈善的继承关系似乎完全被忽略了。虽然有一个带有名称列的组织表,但它与公司或慈善机构都没有主外键关系

Dón*_*nal 2

解决了!

添加以下类src/java(该类不能用 Groovy 编写)

package org.example;

import org.codehaus.groovy.grails.orm.hibernate.cfg.GrailsAnnotationConfiguration;
import org.hibernate.MappingException;
import org.hibernate.mapping.JoinedSubclass;
import org.hibernate.mapping.PersistentClass;
import org.hibernate.mapping.RootClass;

import java.util.Iterator;

public class TablePerSubclassConfiguration extends GrailsAnnotationConfiguration {

    private static final long serialVersionUID = 1;

    private boolean alreadyProcessed = false;

    @Override
    protected void secondPassCompile() throws MappingException {
        super.secondPassCompile();

        if (alreadyProcessed) {
            return;
        }

        for (PersistentClass persistentClass : classes.values()) {
            if (persistentClass instanceof RootClass) {
                RootClass rootClass = (RootClass) persistentClass;

                if (rootClass.hasSubclasses()) {
                    Iterator subclasses = rootClass.getSubclassIterator();

                    while (subclasses.hasNext()) {

                        Object subclass = subclasses.next();

                        // This test ensures that foreign keys will only be created for subclasses that are
                        // mapped using "table per subclass"
                        if (subclass instanceof JoinedSubclass) {
                            JoinedSubclass joinedSubclass = (JoinedSubclass) subclass;
                            joinedSubclass.createForeignKey();
                        }
                    }
                }
            }
        }

        alreadyProcessed = true;
    }
}
Run Code Online (Sandbox Code Playgroud)

然后将DataSource.groovy其设置为配置类

dataSource {
    configClass = 'org.example.TablePerSubclassConfiguration'
    pooled = true
    driverClassName = "org.h2.Driver"
    username = "sa"
    password = ""
    dbCreate = "update"
    url = "jdbc:h2:mem:testDb;MVCC=TRUE;LOCK_TIMEOUT=10000"
}
Run Code Online (Sandbox Code Playgroud)

我已经向 Grails 提交了一个Pull 请求来解决这个问题。该修复已包含在 Grails 2.3.9 中。