使用 JPA 存储 Map<String,String>
Posted
技术标签:
【中文标题】使用 JPA 存储 Map<String,String>【英文标题】:Storing a Map<String,String> using JPA 【发布时间】:2011-03-24 13:06:58 【问题描述】:我想知道是否可以使用注释来使用 JPA2 将 attributes
映射保留在以下类中
public class Example
long id;
// ....
Map<String, String> attributes = new HashMap<String, String>();
// ....
由于我们已经有一个预先存在的生产数据库,所以理想情况下attributes
的值可以映射到以下现有表:
create table example_attributes
example_id bigint,
name varchar(100),
value varchar(100));
【问题讨论】:
【参考方案1】:JPA 2.0 通过@ElementCollection
注释支持原语集合,您可以将其与java.util.Map
集合的支持结合使用。
像这样的东西应该可以工作:
@Entity
public class Example
@Id long id;
// ....
@ElementCollection
@MapKeyColumn(name="name")
@Column(name="value")
@CollectionTable(name="example_attributes", joinColumns=@JoinColumn(name="example_id"))
Map<String, String> attributes = new HashMap<String, String>(); // maps from attribute name to value
另见(在 JPA 2.0 规范中)
2.6 - 可嵌入类和基本类型的集合 2.7 地图集合 10.1.11 - ElementCollection 注释 11.1.29 MapKeyColumn注解【讨论】:
是否有使用 JPA 1 的解决方法?我只找到了Map<String, SomeOtherClass>
的例子
可能值得一提的是,example_attributes
应该有一个复合键 (example_id, name)
- 这是 hbm2ddl 将从上面生成的。
我尝试了上面的示例,但是在尝试持久化实体时,我遇到了一个异常:您必须为此表定义至少一个映射。查询:InsertObjectQuery(null)。有什么提示吗?我创建一个空实体并设置属性映射,然后尝试持久化。
使用 MariaDB,我遇到了Specified key was too long; max key length is 767 bytes
这样做。它尝试创建的主键是 varchar(255) 和 @JoinColumn 的组合,它超出了默认列大小。您需要change your database 或修改您的@MapKeyColumn 以提供长度:@MapKeyColumn(name="name", length=100)
这对我有用!必须确保我的表引用了其他表的 PK。【参考方案2】:
@ElementCollection(fetch = FetchType.LAZY)
@CollectionTable(name = "raw_events_custom", joinColumns = @JoinColumn(name = "raw_event_id"))
@MapKeyColumn(name = "field_key", length = 50)
@Column(name = "field_val", length = 100)
@BatchSize(size = 20)
private Map<String, String> customValues = new HashMap<String, String>();
这是一个关于如何设置地图以控制列名和表名以及字段长度的示例。
【讨论】:
获取类型应该是EAGER以上是关于使用 JPA 存储 Map<String,String>的主要内容,如果未能解决你的问题,请参考以下文章
Spring Data:JPA 存储库 findAll() 返回 *Map 而不是 List?