Я пытаюсь разработать простую веб-приложение 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).
Я знаю только то, что я видел в учебниках, но как Spring узнал о вашем классе PersonController - как он загружается, когда вы видите NPE? – Chris
Пожалуйста, ознакомьтесь с моим изменением в оригинальном посте относительно вашего вопроса. – kolett