Как вы отображаете «карту» в спящем режиме, используя аннотации? - PullRequest
29 голосов
/ 24 февраля 2010

Используя аннотации, как вы отображаете поле в объекте, который представляет собой «Map» (Hashtable) String, для данного объекта? Объект аннотирован, и его экземпляры уже хранятся в базе данных гибернации.

Я нашел синтаксис для определения карты простым ключом и значением как таковой:

<class name="Foo" table="foo">
    ...
    <map role="ages">
         <key column="id"/>
         <index column="name" type="string"/>
         <element column="age" type="string"/>
     </map>
 </class>

И, как ни странно, с сущностью в качестве ключа и простым типом в качестве значения, например:

<class name="Foo" table="foo">
    ...
  <map role="ages">
    <key column="id"/>
    <index-many-to-many column="person_id" 
         class="Person"/>
    <element column="age" type="string"/>
  </map>
</class>
<class name="Person" table="person">
    ...
    <property name="name" column="name" 
         type="string"/>
</class>

Но я не вижу, как это сделать для простого сопоставления ключа и элемента, и я не вижу, как это сделать с помощью аннотаций.

Ответы [ 4 ]

39 голосов
/ 25 февраля 2010

Вы можете просто использовать аннотацию JPA @MapKey (обратите внимание, что аннотация JPA отличается от аннотации Hibernate, Hibernate @MapKey отображает столбец базы данных, содержащий карту ключ, в то время как аннотация JPA отображает свойство, которое будет использоваться в качестве ключа карты).

@javax.persistence.OneToMany(cascade = CascadeType.ALL)
@javax.persistence.MapKey(name = "name")
private Map<String, Person> nameToPerson = new HashMap<String, Person>();
9 голосов
/ 24 февраля 2010
@CollectionOfElements(fetch = FetchType.LAZY)
@JoinTable(name = "JOINTABLE_NAME",
    joinColumns = @JoinColumn(name = "id"))
@MapKey(columns = @Column(name = "name"))
@Column(name = "age")
private Map<String, String> ages = new HashMap<String, String>();
2 голосов
/ 25 января 2016

Я знаю, что этот вопрос очень старый, но, возможно, это может кому-то помочь.

Другая возможность - что-то вроде этого:

@Entity
@Table(name = "PREFERENCE", uniqueConstraints = { @UniqueConstraint(columnNames = { "ID_DOMAIN", "ID_USER", "KEY" })})
public class Preferences {
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    @Column(name = "ID", unique = true, nullable = false)
    private Long id;

    @Column(name = "ID_DOMAIN", unique = false, nullable = false")
    private Long domainId;

    @Column (name = "PREFERENCE_KEY")
    @Enumerated(EnumType.STRING)
    private PreferenceKey key;

    @ManyToOne(fetch = FetchType.LAZY)
    @JoinColumn(name = "ID_USER", referencedColumnName = "ID")
    private User user;
}

and 

@Entity
@Table(name = "USER", uniqueConstraints = { @UniqueConstraint(columnNames = { "ID_DOMAIN", "LOGIN" })})
public class User {
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    @Column(name = "ID", unique = true, nullable = false)
    private Long id;

    @Column(name = "ID_DOMAIN", unique = false, nullable = false")
    private Long domainId;

    // more fields

    @ElementCollection(fetch = FetchType.LAZY)
    @JoinColumns({@JoinColumn(name = "ID_USER", referencedColumnName = "ID"), @JoinColumn(name = "ID_DOMAIN", referencedColumnName = "ID_DOMAIN")})
    @OneToMany(targetEntity = Preferences.class, fetch = FetchType.LAZY)
    @MapKey(name = "key")
    private Map<PreferenceKey, Preferences> preferencesMap;
}

Это создает только две таблицы User и Preferences, обратите внимание, что PreferenceKey уникален для пользователя в домене

1 голос
/ 24 февраля 2010

Вам, вероятно, следует использовать UserType или UserCollectionType. Или вы можете использовать собственный кортеж.

см. документацию ядра hibernate для концепций и документацию аннотаций hibernate для эквивалентного подхода аннотации.

Дайте мне знать, если это не то, что вы просите.

...