mysql - Map<String,String> 集合表的 ElementCollection 未使用 Map 键作为表键的一部分创建

标签 mysql jpa glassfish eclipselink jpa-2.0

编辑:问题已解决,原因未知。

以下实体映射,其中包含 Map<String,String> :

public class Employee {
    @TableGenerator(name="Address_Gen",
            table="ID_GEN",
            pkColumnName="GEN_NAME",
            valueColumnName="GEN_VAL",
            pkColumnValue="Addr_Gen",
            initialValue=10000,
            allocationSize=100)
    @Id @GeneratedValue(generator="Address_Gen")    
    private int id;
    private String name;
    private long salary;

    @ElementCollection
    @CollectionTable(name="EMP_PHONE")
    @MapKeyColumn(name="PHONE_TYPE")
    @Column(name="PHONE_NUM")
    private Map<String, String> phoneNumbers;   
}

结果是一个集合表有一个键(甚至在生成的 DDL 中没有被描述为主键),该键不包含映射键作为键的一部分:

CREATE TABLE `emp_phone` (
  `Employee_ID` int(11) DEFAULT NULL,
  `PHONE_NUM` varchar(255) DEFAULT NULL,
  `PHONE_TYPE` varchar(255) DEFAULT NULL,
  KEY `FK_EMP_PHONE_Employee_ID` (`Employee_ID`),
  CONSTRAINT `FK_EMP_PHONE_Employee_ID` FOREIGN KEY (`Employee_ID`) REFERENCES `employee` (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci

这有错吗?我正在学习的书(ProJPA2)是这样建议的:

Unique tuples in the collection table must be the combination of the key column and the foreign key column that references the source entity instance...[In] the resulting collection table, along with the source EMPLOYEE entity table that it references you can see the primary key constraint on the EMPLOYEE_ID and PHONE_TYPE columns.

我正在使用 eclipse photon、eclipselink 2.5.x、mysql 8.0

提前致谢。

编辑

此后,我使用(几乎)与以前相同的注释创建了一个新项目,但它现在生成了预期的 DDL:

public class Employee {
    @Id
    @GeneratedValue(strategy=GenerationType.IDENTITY) 
    private int id;
    private String name;
    private double salary;
    @ManyToOne  
    private Department department;

    @ElementCollection
    @CollectionTable(name="EMP_PHONE")
    @MapKeyColumn(name="PHONE_TYPE")
    @Column(name="PHONE_NUM")
    private Map<String, String> phoneNumbers = new HashMap<>();

DDL:

CREATE TABLE `emp_phone` (
  `Employee_id` int(11) NOT NULL,
  `PHONE_NUM` varchar(255) DEFAULT NULL,
  `PHONE_TYPE` varchar(255) NOT NULL,
  PRIMARY KEY (`Employee_id`,`PHONE_TYPE`),
  CONSTRAINT `FKq4updxf2ebi3swv5tf3n3jp5h` FOREIGN KEY (`Employee_id`) REFERENCES `employee` (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci

遗憾的是,我没有时间对这两个项目进行取证检查,看看究竟是什么造成了差异。

最佳答案

在这种情况下,EclipseLink 似乎使用了一种更通用的映射方法,只需将映射键列添加到为 List 或 Set 集合类型生成的表中即可。

您可以为此提交增强请求,但这不是一个错误:JPA 没有具体指定生成的表的外观,仅指定用于字段的列。假设 DDL 生成有助于开发原型(prototype),并且您可以根据您的用例优化数据库 - 通过适当添加索引等,但并非所有数据库都以相同的方式工作。

关于mysql - Map<String,String> 集合表的 ElementCollection 未使用 Map 键作为表键的一部分创建,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52190080/

相关文章:

php - mysql如何将mysqli_fetch_array和mysql查询成数组

php - 通过 codeigniter 将帖子数据作为数组从数据库中的网格插入?

java - Hibernate:在 spring jpa 存储库中按字段排序的额外交叉连接

java.lang.ClassCastException : com. example.Entity 无法转换为 com.example.Entity

java - 为什么我无法在 Java Web Socket 应用程序中写入 PrintWriter?

mysql - 通过两个表的 JOIN 获取 DISTINCT 记录

mysql - 数据透视表 - 需要将空值包含为 null

java - 如何在 JPA 配置中设置默认模式名称?

java - URL 将 IIS 和 Glassfish 从端口 8080 重写为 80

web-services - Glassfish 上的安全 Web 服务客户端