对象:实体.ENTITY[id=null]不是已知的实体类型



这是一篇类似于我之前看到的关于这个异常的帖子,但我完全迷失了方向。我还没有使用JPA将实体持久化到数据库中,尽管我已经使用它从表中读取了内容,但没有问题。我的设置是使用Glassfish 3.1.1的Netbeans 7.1,EclipseLink是我的持久性提供者。我有一个非常简单的场景,我只想测试将人名和年龄写入数据库,并使id自动递增。它是一个MySql数据库,包含字段:Id、FirstName和Age。这是我的代码:

Web servlet从html表单中获取名称和年龄:

    @Override
protected void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
    String userPath = request.getServletPath();
    if(userPath.equals("/addUser")){
        //get request parameters from form
        String name = request.getParameter("name");
        String age = request.getParameter("age");
        //set request attributes to be used by forwarded page
        request.setAttribute("name", name);
        request.setAttribute("age", age);
        //create manager class to add person to database
        Manager manager = new Manager();
        manager.addPerson(name, age);
        userPath = "/result";
    }
    // use RequestDispatcher to forward request internally
    String url = "/WEB-INF/view" + userPath + ".jsp";
    try {
        request.getRequestDispatcher(url).forward(request, response);
    } catch (Exception ex) {
        ex.printStackTrace();
    }
}

Manager类,它接受名称和年龄,创建一个person对象并将其持久化。

public class Manager {
private static final String PERSISTENCE_UNIT_NAME = "FormPU";
private static EntityManagerFactory factory;
public Manager() {
}
public void addPerson(String name, String age) {
    factory = Persistence.createEntityManagerFactory(PERSISTENCE_UNIT_NAME);
    EntityManager em = factory.createEntityManager();
    Persons persons = new Persons();
    persons.setName(name);
    persons.setAge(age);
    em.getTransaction().begin();
    em.persist(persons);
    em.getTransaction().commit();
    em.close();
}
}

人员实体类别:

    /**
    *
    * @author esmiala
    */
    @Entity
    @Table(name = "persons")
    @XmlRootElement
    @NamedQueries({
      @NamedQuery(name = "Persons.findAll", query = "SELECT p FROM Persons p"),
      @NamedQuery(name = "Persons.findById", query = "SELECT p FROM Persons p WHERE  
      p.id = :id"),
      @NamedQuery(name = "Persons.findByFirstName", query = "SELECT p FROM Persons p 
      WHERE p.firstName = :firstName"),
      @NamedQuery(name = "Persons.findByAge", query = "SELECT p FROM Persons p WHERE 
      p.age = :age")})
public class Persons implements Serializable {
private static final long serialVersionUID = 1L;
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Basic(optional = false)
@NotNull
@Column(name = "Id")
private Integer id;
@Basic(optional = false)
@NotNull
@Size(min = 1, max = 255)
@Column(name = "FirstName")
private String firstName;
@Basic(optional = false)
@NotNull
@Size(min = 1, max = 255)
@Column(name = "Age")
private String age;
public Persons() {
}
public Persons(Integer id) {
    this.id = id;
}
public Persons(Integer id, String firstName, String age) {
    this.id = id;
    this.firstName = firstName;
    this.age = age;
}
public Integer getId() {
    return id;
}
public void setId(Integer id) {
    this.id = id;
}
public String getFirstName() {
    return firstName;
}
public void setFirstName(String firstName) {
    this.firstName = firstName;
}
public String getAge() {
    return age;
}
public void setAge(String age) {
    this.age = age;
}
@Override
public int hashCode() {
    int hash = 0;
    hash += (id != null ? id.hashCode() : 0);
    return hash;
}
@Override
public boolean equals(Object object) {
    // TODO: Warning - this method won't work in the case the id fields are not set
    if (!(object instanceof Persons)) {
        return false;
    }
    Persons other = (Persons) object;
    if ((this.id == null && other.id != null) || (this.id != null && !this.id.equals(other.id))) {
        return false;
    }
    return true;
}
@Override
public String toString() {
    return "entity.Persons[ id=" + id + " ]";
}

}

Persistence.xml

    <?xml version="1.0" encoding="UTF-8"?>
    <persistence version="2.0" xmlns="http://java.sun.com/xml/ns/persistence" 
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com 
    /xml/ns/persistence/persistence_2_0.xsd">
     <persistence-unit name="FormPU" transaction-type="JTA">
     <provider>org.eclipse.persistence.jpa.PersistenceProvider</provider>
     <jta-data-source>jdbc/form</jta-data-source>
     <exclude-unlisted-classes>false</exclude-unlisted-classes>
    <properties/>
    </persistence-unit>
    </persistence>

注意:我也尝试过将exclude-inlisted-classes标记设置为true,并单独列出类,但这也不起作用。

例外:

    WARNING: StandardWrapperValve[Controller]: PWC1406: Servlet.service() for servlet  
    Controller threw exception
    java.lang.IllegalArgumentException: Object: entity.persons[ id=null ] is not a   
    known entity type.
atorg.eclipse.persistence.internal.sessions.UnitOfWorkImpl.registerNewObject
    ForPersist(UnitOfWorkImpl.java:4141)
atorg.eclipse.persistence.internal.jpa.EntityManagerImpl.   
    persist(EntityManagerImpl.java:368)
at manager.Manager.addPerson(Manager.java:36)
at controller.Controller.doPost(Controller.java:70)
at javax.servlet.http.HttpServlet.service(HttpServlet.java:754)
at javax.servlet.http.HttpServlet.service(HttpServlet.java:847)
at org.apache.catalina.core.StandardWrapper.service(StandardWrapper.java:1523)

等等。任何帮助都将不胜感激!

<exclude-unlisted-classes>不能像您预期的那样工作-persistence.xml中存在此元素会禁用@Entity类的自动发现,无论其中包含什么。

此外,@Entity(name="persons")可能不是您想要的,请使用@Entity @Table (name="persons")

所以你说你可以很好地读取类,但在持久化一个新实例时出错?

你能更新你阅读的对象吗?

您似乎遇到了某种类加载器问题。不知怎的,类路径上有两个类,或者有两个不同的类加载器。传递给持久化的对象来自与JPA使用的类加载器不同的类加载器。您可以检查读取内容和持久化对象的类加载器,以了解它们之间的区别。

你是否重新部署了你的应用程序,或者热部署了?如果您正确关闭/重新启动服务器,它是否工作。请确保在重新部署之前关闭旧的EntityManagerFactory。

关于你的具体问题,试着看看这个链接是否有帮助。

无论如何,实例化EntityManager的方式不是线程安全的。

你可以在这里看到原因。或者,更好的是,您可以使用NetBeans的向导从实体类创建JPA控制器类,并查看它如何注入EntityManager:

@PersistenceContext
private EntityManager em;

另请参阅控制器类(相当于Manager POJO)具有Stateless注释。这是因为您只能在其生命周期由web容器管理的对象中安全地注入EJB(在本例中为EntityManager)(有关访问企业Bean的更多参考,请参阅此处)。

相关内容

最新更新