简体   繁体   中英

JPA OneToOne and ManyToMany between two entities

I asked a question earlier, but think it wasn't clear enough i will elaborate further with the code.
I have a Teacher entity and Department entity
Many Teachers belongs to One Department and
One Department has One Teacher who heads the Department.
Here is how I am implementing it.

@Entity
public class Teacher extends Model {
@Required
public String surname;

@Required
public String othernames;

    ... 

@OneToOne(mappedBy = "deptHead")
public Department headedBy = new Department();

@ManyToOne(cascade=CascadeType.ALL)
public Department dept = new Department();

... 
}

and the Department entity

@Entity
public class Department extends Model {
@Required
public String deptName; 

@OneToMany(mappedBy="dept")
public List<Teacher> teachers = new ArrayList<Teacher>();

@OneToOne
public Teacher deptHead = new Teacher();

    ...

}

I am getting the error below

    @6c4nj8nmg
    Internal Server Error (500) for request GET /

    JPA error
    A JPA error occurred (Unable to build EntityManagerFactory): could not instantiate   test objectmodels.Department

    play.exceptions.JPAException: Unable to build EntityManagerFactory
    at play.db.jpa.JPAPlugin.onApplicationStart(JPAPlugin.java:269)
    at play.plugins.PluginCollection.onApplicationStart(PluginCollection.java:525)
    at play.Play.start(Play.java:526)
    at play.Play.detectChanges(Play.java:630)
    at play.Invoker$Invocation.init(Invoker.java:198)
    at Invocation.HTTP Request(Play!)
    Caused by: org.hibernate.InstantiationException: could not instantiate test objectmodels.Department
    at org.hibernate.engine.UnsavedValueFactory.instantiate(UnsavedValueFactory.java:48)
    at org.hibernate.engine.UnsavedValueFactory.getUnsavedIdentifierValue(UnsavedValueFactory.java:67)
    at org.hibernate.tuple.PropertyFactory.buildIdentifierProperty(PropertyFactory.java:67)
    at org.hibernate.tuple.entity.EntityMetamodel.<init>(EntityMetamodel.java:135)
    at org.hibernate.persister.entity.AbstractEntityPersister.<init>(AbstractEntityPersister.java:485)
    at org.hibernate.persister.entity.SingleTableEntityPersister.<init>(SingleTableEntityPersister.java:133)
    at org.hibernate.persister.PersisterFactory.createClassPersister(PersisterFactory.java:84)
    at org.hibernate.impl.SessionFactoryImpl.<init>(SessionFactoryImpl.java:286)
    at org.hibernate.cfg.Configuration.buildSessionFactory(Configuration.java:1872)
    at org.hibernate.ejb.Ejb3Configuration.buildEntityManagerFactory(Ejb3Configuration.java:906)
    at play.db.jpa.JPAPlugin.onApplicationStart(JPAPlugin.java:267)
    ... 5 more
    Caused by: java.lang.reflect.InvocationTargetException
    at  org.hibernate.engine.UnsavedValueFactory.instantiate(UnsavedValueFactory.java:45)
    ... 15 more
     Caused by: java.lang.StackOverflowError
    at java.lang.Class.searchMethods(Unknown Source)
    at java.lang.Class.getMethod0(Unknown Source)
    at java.lang.Class.getMethod(Unknown Source)
    at      play.classloading.enhancers.PropertiesEnhancer$FieldAccessor.invokeWriteProperty(PropertiesEnhancer.java:268)
    at models.Department.<init>(Department.java:23)
    at models.Teacher.<init>(Teacher.java:47)
    at models.Department.<init>(Department.java:26)
    ...

Needs help on these

This is not a JPA problem, but is caused by recursive instantiation of Teacher / Department .

When you create, or ask JPA to create, an instance of Teacher , the Teacher attempts to instantiate a Department , which instantiates a Teacher ..., to infinity.

Hence you're seeing a StackOverflowError error near the end of that stack trace.

Remove the = new Teacher() and = new Department() expressions from the class definition; depend on and use appropriate setter methods when you create them.

I removed the @oneToOne relationship between Teacher and Department and created a new entity called DepartmentHeads, everything now works fine.
Here is the Results.

@Entity
public class Teacher extends Model {
    @Required
    public String surname;

    @Required
    public String othernames;

    ... 

    @ManyToOne(cascade=CascadeType.ALL)
    public Department dept = new Department();

    ... 
}

@Entity
public class Department extends Model {
    @Required
    public String deptName; 

    @OneToMany(mappedBy="dept")
    public List<Teacher> teachers = new ArrayList<Teacher>();

    ...

}

@Entity
public class DepartmentHead extends Model{
    @OneToOne
    public Teacher teacher = new Teacher();

    @OneToOne
    public Department dept = new Department();  
}

Everything now works fine.

The technical post webpages of this site follow the CC BY-SA 4.0 protocol. If you need to reprint, please indicate the site URL or the original address.Any question please contact:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM