java - 在多对多单向映射中持久化枚举集

标签 java hibernate jpa persistence annotations

我正在使用带有注释的 Hibernate 3.5.2-FINAL 来指定我的持久性映射。我正在努力为应用程序和一组平台之间的关系建模。每个应用程序都可用于一组平台。

从我所做的所有阅读和搜索中,我认为我需要将平台枚举类作为实体持久化,并有一个连接表来表示多对多关系。我希望关系在对象级别是单向的,也就是说,我希望能够获取给定应用程序的平台列表,但我不需要找出给定平台的应用程序列表。

这是我的简化模型类:

@Entity
@Table(name = "TBL_PLATFORM")
public enum Platform {
    Windows,
    Mac,
    Linux,
    Other;

    @Id
    @GeneratedValue
    @Column(name = "ID")
    private Long id = null;

    @Column(name = "NAME")
    private String name;

    private DevicePlatform() {
        this.name = toString();
    }

    // Setters and getters for id and name...
}

@Entity
@Table(name = "TBL_APP")
public class Application extends AbstractEntity implements Serializable {
    private static final long serialVersionUID = 1L;

    @Column(name = "NAME")
    protected String _name;

    @ManyToMany(cascade = javax.persistence.CascadeType.ALL)
    @Cascade({org.hibernate.annotations.CascadeType.SAVE_UPDATE})
    @JoinTable(name = "TBL_APP_PLATFORM", 
              joinColumns = @JoinColumn(name = "APP_ID"),
              inverseJoinColumns = @JoinColumn(name = "PLATFORM_ID"))
    @ElementCollection(targetClass=Platform.class)
    protected Set<Platform> _platforms;

    // Setters and getters...
}

当我运行 Hibernate hbm2ddl 工具时,我看到以下内容(我使用的是 MySQL):

create table TBL_APP_PLATFORM (
    APP_ID bigint not null,
    PLATFORM_ID bigint not null,
    primary key (APP_ID, PLATFORM_ID)
);

相应的外键也从该表创建到应用程序表和平台表。到目前为止一切顺利。

我遇到的一个问题是当我尝试持久化应用程序对象时:

Application newApp = new Application();
newApp.setName("The Test Application");
Set<DevicePlatform> platforms = EnumSet.of(Platform.Windows, Platform.Linux);
newApp.setPlatforms(platforms);
applicationDao.addApplication(newApp);

我希望在 Platform 表中创建适当的行,即为 Windows 和 Linux 创建一行,如果它们不存在的话。然后,应该为新应用程序创建一行,然后在连接表中创建新应用程序与两个平台之间的映射。

我遇到的一个问题是出现以下运行时异常:

2010-06-30 13:18:09,382 6613126-0 ERROR FlushingEventListener Could not synchronize database state with session org.hibernate.TransientObjectException: object references an unsaved transient instance - save the transient instance before flushing: com.example.model.Platform

不知何故,当我尝试持久化应用程序时,平台集没有被持久化。级联注释应该可以解决这个问题,但我不知道出了什么问题。

所以我的问题是:

  1. 有没有更好的方法来模拟我想做的事情,例如使用 Enum 合适吗?
  2. 如果我的模型没问题,如何正确持久化所有对象?

我已经为此苦苦挣扎了好几个小时,并尝试重新创建上面的所有代码,但它可能不完整和/或不准确。我希望有人会指出一些明显的事情!

最佳答案

你应该决定你的Platform是否是一个实体

如果是实体,则不能是enum,因为可能的平台列表存储在数据库中,而不是应用程序中。它应该是带有 @Entity 注释的常规类,并且您将具有正常的多对多关系。

如果不是实体,则不需要TBL_PLATFORM 表,也没有多对多关系。在这种情况下,您可以将一组 Platform 表示为带有位标志的整数字段,或者表示为简单的一对多关系。 JPA 2.0 使用 @ElementCollection 使后一种情况变得简单:

@ElementCollection(targetClass = Platform.class) 
@CollectionTable(name = "TBL_APP_PLATFORM",
    joinColumns = @JoinColumn(name = "APP_ID"))
@Column(name = "PLATFORM_ID")
protected Set<Platform> _platforms; 

-

create table TBL_APP_PLATFORM (   
    APP_ID bigint not null,   
    PLATFORM_ID bigint not null, -- the ordinal number of enum value   
    primary key (APP_ID, PLATFORM_ID)   
);

enum Platform 没有注释。

关于java - 在多对多单向映射中持久化枚举集,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3152787/

相关文章:

java - 从实体创建数据库表

java - Jpa Hibernate Spring Jpa 存储库查询希伯来语值

java - 在打开新的 java swing 之前关闭旧的对话框

java - 如何检测一个 Java 类是被它自己的 main() 调用还是被另一个类调用?

java - Java中如何获取错误原因?

java - JPA @EntityListeners、@PrePersist 和 Spring @RepositoryEventHandler、@HandleBeforeSave 之间的区别

java - 具有多个主机的 RMI

java - 从 hibernate 检索类会抛出 nullpointerException

java - 你觉得有必要把id字段标记为insertable = false和updatable = false吗?

java - jpa只读取 Activity 记录