java.lang.IllegalArgumentException: ожидается отображение IdClass
Я настроил составной первичный ключ для моей сущности Employee
следующее
Employee.java:
@Entity
@Table(name="employee")
@Proxy(lazy=false)
@IdClass(EmployeeId.class)
public class Employee implements Serializable {
private static final long serialVersionUID = 1L;
private EmployeeId employeeId;
private Person person;
private Branch branch;
private boolean isActive;
public Employee() {
}
@EmbeddedId
@AttributeOverrides({
@AttributeOverride(name="person", column = @Column(name="person_id")),
@AttributeOverride(name="branch", column = @Column(name="branch_id"))})
public EmployeeId getEmployeeId() {
return employeeId;
}
public void setEmployeeId(EmployeeId employeeId) {
this.employeeId = employeeId;
}
@ManyToOne(fetch=FetchType.LAZY)
@JoinColumn(name="person_id")
public Person getPerson() {
return person;
}
public void setPerson(Person person) {
this.person = person;
}
@ManyToOne(fetch=FetchType.LAZY)
@JoinColumn(name="branch_id")
public Branch getBranch() {
return branch;
}
public void setBranch(Branch branch) {
this.branch = branch;
}
@Column(name="is_active")
public boolean getIsActive() {
return isActive;
}
public void setIsActive(boolean isActive) {
this.isActive = isActive;
}
}
EmployeeId.java:
@Embeddable
public class EmployeeId implements Serializable {
private static final long serialVersionUID = 1L;
private Person person;
private Branch branch;
public EmployeeId() {
}
public EmployeeId(Person argPerson, Branch argbranch) {
this.person = argPerson;
this.branch = argbranch;
}
@ManyToOne(fetch=FetchType.LAZY)
@JoinColumn(name="person_id", insertable=false, updatable=false)
public Person getPerson() {
return person;
}
public void setPerson(Person person) {
this.person = person;
}
@ManyToOne(fetch=FetchType.LAZY)
@JoinColumn(name="branch_id", insertable=false, updatable=false)
public Branch getBranch() {
return branch;
}
public void setBranch(Branch branch) {
this.branch = branch;
}
}
Я создал SessionFactory
бин с использованием класса org.springframework.orm.hibernate5.LocalSessionFactoryBean
и сопоставил все hbm.xml
как MappingLocations
,
Мой код выдает следующую ошибку:
Caused by: java.lang.IllegalArgumentException: expecting IdClass mapping
at org.hibernate.metamodel.internal.AttributeFactory$3.resolveMember(AttributeFactory.java:971)
at org.hibernate.metamodel.internal.AttributeFactory$5.resolveMember(AttributeFactory.java:1029)
at org.hibernate.metamodel.internal.AttributeFactory.determineAttributeMetadata(AttributeFactory.java:451)
at org.hibernate.metamodel.internal.AttributeFactory.buildIdAttribute(AttributeFactory.java:128)
at org.hibernate.metamodel.internal.MetadataContext.buildIdClassAttributes(MetadataContext.java:337)
at org.hibernate.metamodel.internal.MetadataContext.applyIdMetadata(MetadataContext.java:269)
at org.hibernate.metamodel.internal.MetadataContext.wrapUp(MetadataContext.java:190)
at org.hibernate.metamodel.internal.MetamodelImpl.initialize(MetamodelImpl.java:219)
at org.hibernate.internal.SessionFactoryImpl.<init>(SessionFactoryImpl.java:296)
at org.hibernate.boot.internal.SessionFactoryBuilderImpl.build(SessionFactoryBuilderImpl.java:476)
at org.hibernate.cfg.Configuration.buildSessionFactory(Configuration.java:707)
at org.hibernate.cfg.Configuration.buildSessionFactory(Configuration.java:723)
at org.springframework.orm.hibernate5.LocalSessionFactoryBean.buildSessionFactory(LocalSessionFactoryBean.java:504)
at org.springframework.orm.hibernate5.LocalSessionFactoryBean.afterPropertiesSet(LocalSessionFactoryBean.java:488)
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.invokeInitMethods(AbstractAutowireCapableBeanFac
Как я могу избежать этой ошибки? я использую spring-orm-4.3.1-RELEASE
а также hibernate-core-5.2.0.Final
,
Обновить
Я создал пример проекта, и я получаю следующую ошибку при запуске...
Caused by: org.hibernate.AnnotationException: Property of @IdClass not found in entity sample.domain.Employee: employee
См. Код: https://www.dropbox.com/s/axr8l01iqh0qr29/idclass-using-hibernate5.tar.gz?dl=0
Что я сделал не так? Пожалуйста, предоставьте ваш вклад здесь
7 ответов
Ваша ситуация соответствует главе 2.4.1 Первичные ключи, соответствующие производным идентификаторам спецификации JPA 2.1.
Личность Employee
происходит от личности Person
а также Branch
, Вы не показали код ни одного из них, поэтому я предполагаю, что у них есть простые первичные ключи. В этих отношениях Person
а также Branch
являются "родительскими объектами" и Employee
является "зависимой" сущностью.
Идентификатор Employee
может быть сопоставлен с использованием либо IdClass
или же EmbeddedId
не оба одновременно.
См. Главу 2.4.1.1 Спецификация производных идентификаторов.
Если вы хотите использовать IdClass
, затем:
Имена атрибутов класса id и атрибутов Id класса зависимого объекта должны соответствовать следующим образом:
Id
атрибут в классе сущности и соответствующий атрибут в классе id должны иметь одинаковые имена....
- Если
Id
Атрибут в объекте является отношением "многие к одному" или "один к одному" с родительским объектом, соответствующий атрибут в классе id должен иметь (...) типId
атрибут родительского объекта.
Таким образом, ваши классы будут выглядеть так (геттеры, сеттеры, лишние аннотации и т. Д. Опущены)
@Entity
@IdClass(EmployeeId.class)
public class Employee {
@Id
@ManyToOne
private Person person;
@Id
@ManyToOne
private Branch branch;
}
public class EmployeeId {
private Long person; // Corresponds to the type of Person ID, name matches the name of Employee.person
private Long branch; // Corresponds to the type of Branch ID, name matches the name of Employee.branch
}
Если вы используете EmbeddedId
, затем:
Если зависимый объект использует встроенный идентификатор для представления своего первичного ключа, атрибут во встроенном идентификаторе, соответствующий атрибуту отношения, должен быть того же типа, что и первичный ключ родительского объекта, и должен быть обозначен
MapsId
аннотация применяется к атрибуту отношения.value
элементMapsId
аннотация должна использоваться для указания имени атрибута во встроенном идентификаторе, которому соответствует атрибут отношения.
И код будет выглядеть так:
@Entity
public class Employee {
@EmbeddedId
private EmployeeId id;
@ManyToOne
@MapsId("personId") // Corresponds to the name of EmployeeId.personId
private Person person;
@ManyToOne
@MapsId("branchId") // Corresponds to the name of EmployeeId.branchId
private Branch branch;
}
@Embeddable
public class EmployeeId {
private Long personId; // Corresponds to the type of Person ID
private Long branchId; // Corresponds to the type of Branch ID
}
Составное сопоставление ключей может быть выполнено с использованием IdClass или Embeddable. Если вы хотите использовать IdClass, вы должны аннотировать свои поля в Employee с помощью @Id.
@IdClass(EmployeeId.class)
class Person{
@Id
private Person person;
@Id
private Branch branch;
}
Если вы хотите использовать Embedded в качестве составного ключа, удалите аннотацию @IdClass(EmployeeId.class) из Person. Вам также не нужны поля person и branch в вашем классе Person, потому что они определены в вашем встроенном классе.
Изменить на:
@Entity
@Table(name = "employee")
@Proxy(lazy = false)
@IdClass(EmployeeId.class)
public class Employee implements Serializable {
private static final long serialVersionUID = 1L;
private EmployeeId id;
private Person person;
private Branch branch;
private boolean isActive;
public Employee() {
}
@EmbeddedId
@AttributeOverrides({@AttributeOverride(name = "person", column = @Column(name = "person_id") ),
@AttributeOverride(name = "branch", column = @Column(name = "branch_id") )})
public EmployeeId getId() {
return id;
}
public void setId(EmployeeId id) {
this.id = id;
}
@ManyToOne(fetch = FetchType.LAZY)
@JoinColumn(name = "person_id")
public Person getPerson() {
return person;
}
public void setPerson(Person person) {
this.person = person;
}
@ManyToOne(fetch = FetchType.LAZY)
@JoinColumn(name = "branch_id")
public Branch getBranch() {
return branch;
}
public void setBranch(Branch branch) {
this.branch = branch;
}
@Column(name = "is_active")
public boolean getIsActive() {
return isActive;
}
public void setIsActive(boolean isActive) {
this.isActive = isActive;
}
}
IdClass не должен быть определен как Embeddable -
@Entity
@Table(name="employee")
@IdClass(EmployeeId.class)
public class Employee implements Serializable {
private static final long serialVersionUID = 1L;
@Id
@ManyToOne
private Person person;
@Id
@ManyToOne
private Branch branch;
private boolean isActive;
public Employee() { }
//....
}
А также -
public class EmployeeId implements Serializable {
private static final long serialVersionUID = 1L;
private Person person;
private Branch branch;
public EmployeeId() {}
public EmployeeId(Person argPerson, Branch argbranch) {
this.person = argPerson;
this.branch = argbranch;
}
}
Прочитайте ваш комментарий. Могу ли я предложить вам сопоставить Employee с person_id и branch_id, а не с объектами JPA Person и Branch? Это позволит нам проверить правильность вашей конфигурации hbm. Я также рекомендую опубликовать ваш конфиг hbm, так как я думаю, что в этой проблеме отсутствует информация
Таким образом, таблица будет похожа на -
@Entity
@Table(name="employee")
@IdClass(EmployeeId.class)
public class Employee implements Serializable {
private static final long serialVersionUID = 1L;
@Id
private Long personId;
@Id
private Long branchId;
private boolean isActive;
public Employee() { }
//....
}
А также -
А также -
public class EmployeeId implements Serializable {
private static final long serialVersionUID = 1L;
private Long personId;
private Long branchId;
public EmployeeId() {}
public EmployeeId(Person argPerson, Branch argbranch) {
this.person = argPerson;
this.branch = argbranch;
}
}
Эта ссылка может помочь вам JPA - EmbeddedId с @ManytoOne
Отображения отношений, определенные в классе встроенного идентификатора, не поддерживаются. Затем необходимо изменить класс встроенного идентификатора следующим образом.
@Embeddable
public class EmployeeId implements Serializable {
private static final long serialVersionUID = 1L;
private Long personId;
private Long branchId;
public EmployeeId() {
}
public EmployeeId(Long argPerson, Long argbranch) {
this.personId = argPerson;
this.branchId = argbranch;
}
@Column(name = "person_id")
public Long getPersonId() {
return personId;
}
public void setPersonId(Long personId) {
this.personId = personId;
}
@Column(name = "branch_id")
public Long getBranchId() {
return branchId;
}
public void setBranchId(Long branchId) {
this.branchId = branchId;
}
}
Упоминание @IdClass
аннотация с классом, который содержит ID
, Проверьте ответ на этот пост
JPA Составной первичный ключ
Задает класс составного первичного ключа, который сопоставляется нескольким полям или свойствам сущности.
Имена полей или свойств в классе первичного ключа и полях первичного ключа или свойствах сущности должны соответствовать, и их типы должны быть одинаковыми.
Ответ здесь. прочитайте описание для вас. введите описание ссылки здесь
(Образец кода)
@Entity
@Table(name = "EMP_PROJECT")
@IdClass(ProjectAssignmentId.class)
public class ProjectAssignment {
@Id
@Column(name = "EMP_ID", insertable = false, updatable = false)
private int empId;
@Id
@Column(name = "PROJECT_ID", insertable = false, updatable = false)
private int projectId;
@ManyToOne
@JoinColumn(name = "EMP_ID")
Professor employee;
@ManyToOne
@JoinColumn(name = "PROJECT_ID")
Project project;
....
}
public class ProjectAssignmentId implements Serializable {
private int empId;
private int projectId;
...
}