在 JPA (EclipseLink) Spring 项目中的 Jersey 控制器中,EntityManager 为空

EntityManager is null in Jersey controller in Spring project with JPA (EclipseLink)

我正在尝试使用 Jersey (JAX-RS) 开发一个简单的 Spring Web 应用程序,部署在 Tomcat 容器中。这些实体使用 JPA 和 EclipseLink 提供程序进行管理,并存储在 MySQL 数据库中。我不使用 EJB。

我正在尝试通过 Spring 注入 EntityManager,但是当我想从数据库中获取(或持久化)一个实体时,我收到一个 NullPointerException,表明 EntityManager 为空。

我花了很多时间寻找解决方案并尝试了我在教程和主题中找到的所有代码,但得到了相同的结果。

我已经在项目中包含了所有必需的依赖项。如果我手动创建一个 EntityManagerFactory,然后在 getPerson 中创建一个 EntityManager,它会起作用,但我认为这不是正确的方法。 此外,当我启动该服务时,我可以在控制台输出日志中看到 Spring root WebApplicationContext 已初始化,bean 定义已加载,持久性单元 'defaultPU' 的 JPA 容器 EntityManagerFactory 已构建。

我显然遗漏了一些东西,你能帮我怎么做吗? 为什么 EntityManager 为空,我应该如何注入它才能在 PersonController 中使用它?

Person.java:

package TestSpringApp;

import javax.persistence.Entity;
import javax.persistence.Id;

@Entity
public class Person {
    @Id
    private int id;
    private String name;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

PersonController.java:

package TestSpringApp;

import javax.persistence.EntityManager;
import javax.persistence.PersistenceContext;
import javax.ws.rs.GET;
import javax.ws.rs.Path;

@Component
@Path("/person")
public class PersonController {

    @PersistenceContext
    EntityManager entityManager;

    @GET
    @Transactional
    public String getPerson() {
        Person p = entityManager.find(Person.class, 0);
        return p.getName();
    }

    public void setEntityManager(EntityManager em) {
        this.entityManager = em;
    }

    public EntityManager getEntityManager() {
        return this.entityManager;
    }
}

beans.xml:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:tx="http://www.springframework.org/schema/tx"
       xsi:schemaLocation=
        "http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd
        http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">

    <context:component-scan base-package="TestSpringApp"/>

    <tx:annotation-driven />

    <context:annotation-config/>

    <bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
        <property name="jpaVendorAdapter" ref="jpaAdapter" />
        <property name="persistenceUnitName" value="defaultPU"/>
    </bean>

    <bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
        <property name="entityManagerFactory" ref="entityManagerFactory" />
    </bean>

    <bean id="jpaAdapter" class="org.springframework.orm.jpa.vendor.EclipseLinkJpaVendorAdapter">
        <property name="database" value="MYSQL" />
        <property name="showSql" value="true" />
    </bean>

</beans>

persistence.xml:

<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="defaultPU" transaction-type="RESOURCE_LOCAL">
        <provider>org.eclipse.persistence.jpa.PersistenceProvider</provider>
        <class>TestSpringApp.Person</class>

        <properties>
<property name="javax.persistence.jdbc.url" value="jdbc:mysql://localhost:3306/test"/>
            <property name="javax.persistence.jdbc.driver" value="com.mysql.jdbc.Driver"/>
            <property name="javax.persistence.jdbc.user" value="root"/>
            <property name="javax.persistence.jdbc.password" value="root"/>
            <property name="eclipselink.ddl-generation" value="create-tables" />
            <property name="eclipselink.ddl-generation.output-mode" value="database" />
            <property name="eclipselink.weaving" value="false"/>
        </properties>
    </persistence-unit>
</persistence>

web.xml:

<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xmlns="http://java.sun.com/xml/ns/javaee"
         xsi:schemaLocation="http://java.sun.com/xml/ns/javaee
         http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
         id="WebApp_ID" version="3.0">

    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>classpath:beans.xml</param-value>
    </context-param>

    <listener>
        <listener-class>
            org.springframework.web.context.ContextLoaderListener
        </listener-class>
    </listener>

    <display-name>testspring</display-name>

    <servlet>
        <servlet-name>/</servlet-name>
        <servlet-class>org.glassfish.jersey.servlet.ServletContainer</servlet-class>
        <init-param>
            <param-name>jersey.config.server.provider.packages</param-name>
            <param-value>TestSpringApp</param-value>
        </init-param>
        <load-on-startup>1</load-on-startup>
    </servlet>

    <servlet-mapping>
        <servlet-name>/</servlet-name>
        <url-pattern>/rest/*</url-pattern>
    </servlet-mapping>

</web-app>

编辑:

直到现在,我一直认为 context:component-scan 或 beans.xml 中的 context:annotation-config 负责让 Spring 知道 PersonController,但是现在我检查它,我发现只有当我将 @PersistenceContext 放在 setEntityManager 而不是 EntityManager 本身上并用 @Component 注释 PersonController class 时,才会调用 setEntityManager。这样我可以看到 setEntityManager 在 Spring bean 初始化时被调用,它的值是 "Shared EntityManager proxy for target factory [org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean@15bd577]".
我的问题仍然存在,因为当我执行 http://localhost:8080/rest/person 请求时,entityManager 在行 entityManager.find(Person.class, 0).

中仍然为 null

阅读 Jersey 文档后,我发现 this chapter 关于 Spring DI 支持。

向项目添加 jersey-spring3spring-bridge 依赖项解决了问题。