1

这似乎应该相当简单,但我很难想出一个优雅的解决方案。

让我们使用一个基本文件系统的示例,它由两个带 hibernate 注释的类FileFolder组成。

我们将两个类的共同属性抽象为FileSystemObject接口:

public interface FileSystemObject {
    public String getName();
    public URI getWhere();
}

还有两个类:

文件

@Entity
@Table(name = "FILE")
public class File implements FileSystemObject, Serializable {

    private static final long serialVersionUID = 1L;

    @Id
    @Column(name = "FILE_ID")
    @GeneratedValue(strategy = GenerationType.AUTO)
    private long id;

    @Column(name = "NAME", nullable = false)
    @Index(name = "FILE_NAME", columnNames={"NAME"})
    private String name;

    @Column(nullable = false)
    private URI where;

    public File() {}

    public File(String name, URI where) {
        this.name = name;
        this.where = where;
    }

    public long getId() {
        return id;
    }

    public void setId(long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public URI getWhere() {
        return where;
    }

    public void setWhere(URI where) {
        this.where = where;
    }
}

文件夹

@Entity
@Table(name = "FOLDER")
public class Folder implements FileSystemObject, Serializable {

    private static final long serialVersionUID = 2L;

    @Id
    @Column(name = "FOLDER_ID")
    @GeneratedValue(strategy = GenerationType.AUTO)
    private long id;

    @Column(name = "NAME", nullable = false)
    @Index(name = "FOLDER_NAME", columnNames={"NAME"})
    private String name;

    @Column(nullable = false)
    private URI where;

    @CollectionOfElements
    @JoinTable(name = "FOLDER_CONTENTS",
               joinColumns = @JoinColumn(name = "FOLDER_ID"))
    private List<FileSystemObject> contents;

    public Folder() {}

    public Folder(String name, URI where, List<FileSystemObject> contents) {
        this.name = name;
        this.where = where;
        this.contents = contents;
    }

    public long getId() {
        return id;
    }

    public void setId(long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public URI getWhere() {
        return where;
    }

    public void setWhere(URI where) {
        this.where = where;
    }

    public List<FileSystemObject> getContents() {
        return contents;
    }

    public void setContents(List<FileSystemObject> contents) {
        this.contents = contents;
    }
}

从逻辑上讲,Folder的内容可以是FileFolder ,因此内容 List 的类型为FileSystemObject是有意义的。

现在,假设一切都在 pom.xml 中正确设置,当我尝试生成一些模式时:

mvn hibernate3:hbm2ddl

它抛出以下错误:

Failed to execute goal org.codehaus.mojo:hibernate3-maven-plugin:2.0:hbm2ddl (generate-ddl) on
project foo: Execution generate-ddl of goal org.codehaus.mojo:hibernate3-maven
plugin:2.0:hbm2ddl failed: Could not determine type for: com.foo.data.FileSystemObject, for
columns: [org.hibernate.mapping.Column(element)] -> [Help 1]

希望有人可以对此有所了解!

4

1 回答 1

1

该类型可能有一个接口,但它不会按照您配置实体的方式工作。为了将两种实体类型组合在同一个集合中,它们需要共享一个公共超类型,该超类型本身就是一个实体。(也就是说,您需要像 FileSystemObject 这样的东西作为它们都继承自的公共超类,在该级别定义 ID。)

问题是,考虑这个查询:

select c from Folder f, f.contents c where f.name = 'FOLDER' and c.id = 3;

如果文件夹和文件的 id 为 3,它怎么知道你想要什么?这就是为什么如果它们要在同一个集合中,它们需要共享一个公共超类。

于 2012-10-10T01:35:22.003 回答