持久化上下文 EntityManager 注入 NullPointerException

46

我有一个变量包含以下内容:

META-INF/MANIFEST.MF
WEB-INF/classes/META-INF/persistence.xml
WEB-INF/classes/com/test/service/TestServlet.class
WEB-INF/classes/com/test/service/TestEntity.class
WEB-INF/classes/jndi.properties
WEB-INF/classes/postgresql-ds.xml
WEB-INF/jboss-web.xml
WEB-INF/web.xml
index.jsp

persistence.xml:

<?xml version="1.0" encoding="UTF-8"?>
<persistence 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_1_0.xsd" version="1.0">
    <persistence-unit name="test">
        <provider>org.hibernate.ejb.HibernatePersistence</provider>
        <jta-data-source>java:/TestDS</jta-data-source>

        <properties>
            <property name="hibernate.dialect" value="org.hibernate.dialect.PostgreSQLDialect"/>
            <property name="hibernate.hbm2ddl.auto" value="update" />
            <property name="hibernate.show_sql" value="true" />
        </properties>
    </persistence-unit>
</persistence>

web.xml:

<!DOCTYPE web-app PUBLIC
 "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
 "http://java.sun.com/dtd/web-app_2_3.dtd" >

<web-app>
  <display-name>Test Web Application</display-name>

    <context-param>
        <param-name>resteasy.scan</param-name>
        <param-value>true</param-value>
    </context-param>
    <listener>
        <listener-class>org.jboss.resteasy.plugins.server.servlet.ResteasyBootstrap</listener-class>
    </listener>
    <servlet>
        <servlet-name>Resteasy</servlet-name>
        <servlet-class>org.jboss.resteasy.plugins.server.servlet.HttpServletDispatcher</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>Resteasy</servlet-name>
        <url-pattern>/service/*</url-pattern>
    </servlet-mapping>

    <resource-ref>
        <res-ref-name>TestDS</res-ref-name>
        <res-type>javax.sql.DataSource</res-type>
        <res-auth>Container</res-auth>
        <res-sharing-scope>Shareable</res-sharing-scope>
    </resource-ref>
</web-app>

我的TestServlet类如下:

package com.test.service;

import java.util.*;
import javax.persistence.*;
import javax.ws.rs.*;

@Path("/service")
public class TestService {

    @PersistenceContext(unitName = "test")
    private EntityManager em;

    @GET
    @Path("/get")
    @Produces("application/json")
    public List get() {
        return em.createQuery("from TestEntity").getResultList();
    }
}
当调用get()方法时,我遇到了NullPointerException;EntityManager没有被注入。你有什么建议,我可能错过了什么或者如何诊断它?服务器日志中几乎没有任何内容。
我确定在没有jboss-web.xml或web.xml中的datasource条目的情况下可以使其工作。我也将ds.xml单独部署到了deploy目录中,它肯定已被加载 - 我可以在JMX控制台中看到它。
尝试使用JBoss 4.2.3和一个6.0版本构建得到了同样的结果。

在代码段中发布XML存在问题吗?! - rich
你需要转义符号 < 和 >,分别用 < 和 > 来表示。 - adrianboimvaser
4
我从未在实践中尝试过,因此不要依赖我的答案,但是否真的可能在一个已声明为运行为Servlet 2.3(J2EE 1.3)而非Servlet 2.5(Java EE 5,在JPA首次引入时)的Web应用程序上运行JPA? - BalusC
值得一试,本来就想做,但更新到2.5声明并没有改变行为。 - rich
3个回答

82

实体管理器只能在运行在事务内的类中注入。换句话说,它只能在 EJB 中注入。其他类必须使用 EntityManagerFactory 创建和销毁 EntityManager。

由于您的 TestService 不是 EJB,因此注释 @PersistenceContext 将被忽略。不仅如此,在 JavaEE 5 中,无法在 JAX-RS 服务中注入 EntityManager 或 EntityManagerFactory。您需要使用 JavaEE 6 服务器(JBoss 6、Glassfish 3 等)。

这里是注入 EntityManagerFactory 的示例:

package com.test.service;

import java.util.*;
import javax.persistence.*;
import javax.ws.rs.*;

@Path("/service")
public class TestService {

    @PersistenceUnit(unitName = "test")
    private EntityManagerFactory entityManagerFactory;

    @GET
    @Path("/get")
    @Produces("application/json")
    public List get() {
        EntityManager entityManager = entityManagerFactory.createEntityManager();
        try {
            return entityManager.createQuery("from TestEntity").getResultList();
        } finally {
            entityManager.close();
        }
    }
}

如果您使用JavaEE 6服务器,最简单的方法是将您的服务声明为EJB 3.1。

相关问题: 将EJB注入JAX-RS(RESTful服务)


1
谢谢,虽然问题类似,但我在这一行代码上遇到了NPE:<pre><code> EntityManager entityManager = entityManagerFactory.createEntityManager(); </code></pre> - rich
谢谢,这对我很有帮助!我正在使用CXF 2.7.5、Hibernate 4.x、Spring 3.x和JPA 2.x。 - Kode Charlie
请告诉我为什么我无法在Tomcat 7.0.27中将实体管理器注入到无状态bean @PersistenceContext(unitName = "Tester2PU") private EntityManager em; 中,它是null。谢谢,Senthil。 - user1503117
@user1503117 你好,Tomcat不是EJB容器。请尝试使用TomEE、GlassFish或JBoss AS。 - Evan Hu
1
@Andre: "实体管理器只能在运行在事务内的类中注入"。我不认为这是绝对正确的。您可以注入一个扩展的实体管理器,并在非事务性方法中使用它,然后调用事务性方法并加入事务(自动或通过调用EntityManager@joinTransaction())。 - Mike Argyriou
这还是真的吗? - Jin Kwon

2
如果组件是EJB,则注入EM不应该有问题。
但是,在JBoss 5中,JAX-RS集成不太好。如果您有一个EJB,则无法使用扫描,必须手动列出context-param resteasy.jndi.resource。如果您仍然在扫描,请注意,Resteasy将扫描资源类并将其注册为普通的JAX-RS服务并处理生命周期。
这很可能是问题所在。

0
如果您的实体类中有任何NamedQueries,请检查编译错误的堆栈跟踪。无法编译的格式不正确的查询可能会导致无法加载持久性上下文。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接