2016-09-08 6 views
0

Я пытаюсь разработать простую веб-приложение Spring с использованием Jersey (JAX-RS), развернутую в контейнере Tomcat. Объекты управляются с использованием JPA с провайдером EclipseLink и хранятся в базе данных MySQL. Я не использую EJB.EntityManager имеет значение null в контроллере Джерси в проекте Spring с JPA (EclipseLink)

Я пытаюсь ввести EntityManager через Spring, однако, когда я хочу получить (или сохранить) сущность из БД, я получаю исключение NullPointerException, показывающее, что EntityManager имеет значение null.

Я потратил много часов, пытаясь найти решение и пробовал каждый код, который я нашел в учебниках и потоках, но получил тот же результат.

Я включил все необходимые зависимости в проект. Если я вручную создаю EntityManagerFactory, а затем EntityManager в getPerson, он работает, но я не думаю, что это должно быть правильно. Кроме того, когда я запускаю службу, я могу видеть в журналах вывода консоли, что Spring root WebApplicationContext инициализирован, загружаются определения бобов и встроен контейнер JPA EntityManagerFactory для единицы по умолчанию «defaultPU».

У меня явно нет чего-то, не могли бы вы помочь мне, как я могу сделать эту работу? Почему EntityManager null и как его следует вводить, чтобы иметь возможность использовать его в 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> 

EDIT:

До сих пор я думал, что контекст: компонент сканирование или контекст: аннотация-конфигурация в beans.xml заботилась сделать Spring осознает PersonController , однако теперь я проверил его, и я узнал, что setEntityManager вызывается только в том случае, если я помещаю @PersistenceContext в setEntityManager вместо самого EntityManager и аннотирую класс PersonController с @Component. Таким образом, я вижу, что setEntityManager вызывается при инициализации Spring bean, а его значение - «Shared EntityManager proxy для целевой фабрики [org[email protected]15bd577]».
Моя проблема все еще встречается, потому что когда я делаю запрос http://localhost:8080/rest/person, entityManager все еще имеет значение null в строке entityManager.find (Person.class, 0).

+0

Я знаю только то, что я видел в учебниках, но как Spring узнал о вашем классе PersonController - как он загружается, когда вы видите NPE? – Chris

+0

Пожалуйста, ознакомьтесь с моим изменением в оригинальном посте относительно вашего вопроса. – kolett

ответ

0

После прочтения документов в Джерси я нашел this chapter в поддержку Spring DI в Джерси.

Добавление jersey-spring3 и spring-bridge dependecies к проекту решила проблему.