PersistenceContext EntityManager 注入 NullPointerException

Posted

技术标签:

【中文标题】PersistenceContext EntityManager 注入 NullPointerException【英文标题】:PersistenceContext EntityManager injection NullPointerException 【发布时间】:2011-06-10 03:09: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 中的数据源条目的情况下可以正常工作。我也已经将 ds.xml 单独部署到了 deploy 目录,这肯定已经被选中了 - 我可以在 JMX 控制台中看到它。

尝试使用 JBoss 4.2.3 和 6.0 构建,结果相同。

【问题讨论】:

猜想在代码段中发布 XML 有问题?! 你必须逃避 ->< > 我从来没有在实践中尝试过,所以不要把我钉在上面,但是真的有可能在一个被声明为 Servlet 2.3 (J2EE 1.3) 的 webapp 上运行 JPA而不是 Servlet 2.5(Java EE 5,首次引入 JPA 时)? 值得一试,无论如何都想做,但更新到 2.5 声明并没有改变行为。 【参考方案1】:

实体管理器只能注入在事务内部运行的类中。换句话说,它只能注入到 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。

相关问题: Inject an EJB into JAX-RS (RESTful service)

【讨论】:

谢谢,类似的问题虽然我在这一行得到了一个 NPE:
 EntityManager entityManager = entityManagerFactory.createEntityManager(); 
谢谢,这对我有用!我正在使用 CXF 2.7.5、Hibernate 4.x、Spring 3.x 和 JPA 2.x。 请让我知道为什么我无法在无状态 bean 中注入实体管理器 @PersistenceContext(unitName = "Tester2PU") private EntityManager em;在 tomcat 7.0.27 中为空。谢谢森希尔 @user1503117 嗨,tomcat 不是 EJB 容器。请尝试 TomEE、GlassFish 或 JBoss AS。 @Andre:“实体管理器只能注入在事务内部运行的类中”。我不认为这是绝对正确的。您可以注入 EXTENDED 实体管理器并在非事务方法中使用它,然后调用事务方法并加入事务(自动或通过调用 EntityManager@joinTransaction())。【参考方案2】:

如果组件是 EJB,那么注入 EM 应该没有问题。

但是....在 JBoss 5 中,JAX-RS 集成不是很好。如果您有 EJB,则不能使用扫描,并且必须在上下文参数 resteasy.jndi.resource 中手动列出。如果您仍然在扫描,Resteasy 将扫描资源类并将其注册为 vanilla JAX-RS 服务并处理生命周期。

这可能是问题所在。

【讨论】:

【参考方案3】:

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

【讨论】:

以上是关于PersistenceContext EntityManager 注入 NullPointerException的主要内容,如果未能解决你的问题,请参考以下文章

Java CDI @PersistenceContext 和线程安全

如何在@FacesConverter 中注入@EJB、@PersistenceContext、@Inject、@Autowired 等?

使用 @PersistenceContext(type=PersistenceContextType.EXTENDED) 时避免 1 级缓存问题

@ApplicationScoped CDI bean 和 @PersistenceContext - 这安全吗?

Spring Boot 中的 PersistenceContext 生命周期

为啥我们必须在扩展的 PersistenceContext 中手动 flush() EntityManager?