javax.persistence.TransactionRequiredException: 当前线程中没有可用的带有实际事务的EntityManager。

10 浏览
0 Comments

javax.persistence.TransactionRequiredException: 当前线程中没有可用的带有实际事务的EntityManager。

我使用Hibernate创建了我的第一个Spring MVC项目。我的DAO层使用JPA EntityManager与数据库进行交互。

GenericDao.java:

@Repository

public abstract class GenericDao implements GeneralDao {

private Class className;

public GenericDao(Class className) {

this.className = className;

}

@PersistenceContext

private EntityManager entityManager;

public EntityManager getEntityManager() {

return entityManager;

}

@Override

public void add(T object) {

try {

getEntityManager().persist(object);

} catch (HibernateException e) {

throw new DaoException(ErrorMessage.ADD_ENTITY_FAIL, e);

}

}

@Override

public void update(T object) {

try {

getEntityManager().merge(object);

} catch (HibernateException e) {

throw new DaoException(ErrorMessage.UPDATE_ENTITY_FAIL, e);

}

}

@Override

public void remove(T object) {

try {

getEntityManager().remove(object);

} catch (HibernateException e) {

throw new DaoException(ErrorMessage.REMOVE_ENTITY_FAIL, e);

}

}

@Override

public T getById(int id) {

try {

return getEntityManager().find(this.className, id);

} catch (HibernateException e) {

throw new DaoException(ErrorMessage.GET_BY_ID_ENTITY_FAIL, e);

}

}

public abstract List getAll() throws DaoException;

}

GenericService.java

@Service

public abstract class GenericService implements GeneralService {

private static Logger logger = Logger.getLogger(GenericService.class);

@Autowired

private GenericDao dao;

@Transactional

@Override

public void add(T object) throws ServiceException {

try {

dao.add(object);

} catch (DaoException e) {

logger.debug(e);

throw new ServiceException(e.getMessage());

}

}

@Transactional

@Override

public void update(T object) throws ServiceException {

try {

dao.update(object);

} catch (DaoException e) {

logger.debug(e);

throw new ServiceException(e.getMessage());

}

}

@Transactional

@Override

public void remove(T object) throws ServiceException {

try {

dao.remove(object);

} catch (DaoException e) {

logger.debug(e);

throw new ServiceException(e.getMessage());

}

}

@Transactional(readOnly = true)

@Override

public T getById(int id) throws ServiceException {

try {

return dao.getById(id);

} catch (DaoException e) {

logger.debug(e);

throw new ServiceException(e.getMessage());

}

}

@Transactional(readOnly = true)

@Override

public List getAll() throws ServiceException {

try {

return dao.getAll();

} catch (DaoException e) {

logger.debug(e);

throw new ServiceException(e.getMessage());

}

}

}

UserServiceImpl.java:

@Service

public class UserServiceImpl extends GenericService implements UserService {

private static Logger logger = Logger.getLogger(UserServiceImpl.class);

@Autowired

private UserDao userDao;

@Transactional

@Override

public String checkUser(String userLogin, String userPassword) throws ServiceException {

String namePage = "errorAuthorization";

List userList;

try {

userList = userDao.getByLoginAndPassword(userLogin, userPassword);

} catch (DaoException e) {

logger.debug(e);

throw new ServiceException(e.getMessage());

}

if(userList.size() != 0) {

return UserRoleChecker.defineUserPage(userList.get(0));

}

return namePage;

}

@Transactional

@Override

public void addUser(String userLogin, String userPassword, String userMail) throws ServiceException {

Role role = new Role(0L, RoleType.USER);

User user = new User(0L, userLogin, userPassword, userMail, role);

add(user);

}

}

UserController.java:

@Controller

public class UserController {

private static String className = UserController.class.getName();

private static Logger logger = Logger.getLogger(UserController.class.getName());

@Autowired

private UserService userService;

@RequestMapping(value = "/check_user", method = RequestMethod.POST)

public ModelAndView authorizationUser(HttpServletRequest request, HttpServletResponse response) {

ModelAndView modelAndView = new ModelAndView();

String returnPage;

try {

returnPage = userService.checkUser(request.getParameter(RequestParameter.USER_LOGIN), request.getParameter(RequestParameter.USER_PASSWORD));

} catch (ServiceException e) {

logger.debug(e);

returnPage = ErrorHandler.returnErrorPage(e.getMessage(), className);

}

modelAndView.setViewName(returnPage);

return modelAndView;

}

@RequestMapping(value = "/add_user", method = RequestMethod.POST)

public ModelAndView registrationUser(HttpServletRequest request, HttpServletResponse response) {

ModelAndView modelAndView = new ModelAndView();

String returnPage = Page.SUCCESSFUL_REGISTRATION;

try {

userService.addUser(request.getParameter(RequestParameter.USER_LOGIN), request.getParameter(RequestParameter.USER_PASSWORD), request.getParameter(RequestParameter.USER_MAIL));

} catch (ServiceException e) {

logger.debug(e);

returnPage = ErrorHandler.returnErrorPage(e.getMessage(), className);

}

modelAndView.setViewName(returnPage);

return modelAndView;

}

}

root-context.xml:

xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xmlns:p="http://www.springframework.org/schema/p"

xmlns:c="http://www.springframework.org/schema/c"

xmlns:context="http://www.springframework.org/schema/context"

xmlns:util="http://www.springframework.org/schema/util"

xmlns:jdbc="http://www.springframework.org/schema/jdbc" xmlns:tx="http://www.springframework.org/schema/tx"

xsi:schemaLocation="http://www.springframework.org/schema/jdbc http://www.springframework.org/schema/jdbc/spring-jdbc-3.1.xsd

http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd

http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util.xsd

http://www.springframework.org/schema/jdbc http://www.springframework.org/schema/jdbc/spring-jdbc.xsd

http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx.xsd">

org.hibernate.dialect.MySQLDialect

true

true

2

true

update

Logs:

org.springframework.web.servlet.FrameworkServlet 2017-05-10 22:23:59,107 DEBUG - 无法完成请求

javax.persistence.TransactionRequiredException: 当前线程没有可用的实际事务的EntityManager - 无法可靠地处理'persist'调用

at org.springframework.orm.jpa.SharedEntityManagerCreator$SharedEntityManagerInvocationHandler.invoke(SharedEntityManagerCreator.java:282)

at com.sun.proxy.$Proxy27.persist(Unknown Source)

at by.netcracker.artemyev.dao.GenericDao.add(GenericDao.java:35)

at by.netcracker.artemyev.service.GenericService.add(GenericService.java:24)

at by.netcracker.artemyev.service.impl.UserServiceImpl.addUser(UserServiceImpl.java:48)

at by.netcracker.artemyev.web.UserController.registrationUser(UserController.java:45)

at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)

at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)

at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)

at java.lang.reflect.Method.invoke(Method.java:498)

at org.springframework.web.method.support.InvocableHandlerMethod.doInvoke(InvocableHandlerMethod.java:205)

at org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:133)

at org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:97)

at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandlerMethod(RequestMappingHandlerAdapter.java:827)

at org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:738)

at org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:85)

at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:963)

at org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:897)

at org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:970)

at org.springframework.web.servlet.FrameworkServlet.doPost(FrameworkServlet.java:872)

at javax.servlet.http.HttpServlet.service(HttpServlet.java:661)

at org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:846)

at javax.servlet.http.HttpServlet.service(HttpServlet.java:742)

at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:230)

at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:165)

at org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:53)

at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:192)

at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:165)

at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:199)

at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:96)

at org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:475)

at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:140)

at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:80)

at org.apache.catalina.valves.AbstractAccessLogValve.invoke(AbstractAccessLogValve.java:624)

at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:87)

at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:341)

at org.apache.coyote.http11.Http11Processor.service(Http11Processor.java:495)

at org.apache.coyote.AbstractProcessorLight.process(AbstractProcessorLight.java:66)

at org.apache.coyote.AbstractProtocol$ConnectionHandler.process(AbstractProtocol.java:767)

at org.apache.tomcat.util.net.NioEndpoint$SocketProcessor.doRun(NioEndpoint.java:1354)

at org.apache.tomcat.util.net.SocketProcessorBase.run(SocketProcessorBase.java:49)

at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)

at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)

at org.apache.tomcat.util.threads.TaskThread$WrappingRunnable.run(TaskThread.java:61)

at java.lang.Thread.run(Thread.java:745)

为什么我会有这个问题,如何解决它?

0
0 Comments

在使用JPA进行数据库操作时,有时会出现(javax.persistence.TransactionRequiredException: No EntityManager with actual transaction available for current thread)这个错误。这个问题的出现是由于当前线程中没有可用的实际事务的EntityManager引起的。

解决这个问题的方法是在控制器或服务的上方添加@Transactional注解。代码示例如下:

@Transactional
public class MyController {
    // controller methods
}

添加@Transactional注解可以确保在当前线程中存在可用的事务,以解决(javax.persistence.TransactionRequiredException: No EntityManager with actual transaction available for current thread)的问题。

这种问题通常发生在没有正确配置事务管理器的情况下。使用@Transactional注解可以告诉Spring框架,该方法需要在事务中执行,并且将自动创建一个事务管理器。

总结一下,通过在控制器或服务类的上方添加@Transactional注解,可以解决(javax.persistence.TransactionRequiredException: No EntityManager with actual transaction available for current thread)这个问题,确保在当前线程中存在可用的事务。

0
0 Comments

在维护一个遗留代码时,我遇到了这个问题,最后我从复杂的注解中意识到了问题所在。

它需要在应用程序级别添加

@EnableTransactionManagement

。否则,

@Transactional

将无法工作。

这个问题的出现原因是由于事务管理器没有正确配置。在使用Spring框架进行数据库操作时,我们通常使用

@Transactional

注解来开启事务。然而,在某些情况下,如果没有正确配置事务管理器,就会出现这个问题。

解决这个问题的方法是在应用程序的配置类上添加

@EnableTransactionManagement

注解。这个注解告诉Spring框架启用事务管理器,并在需要的时候自动创建和管理事务。

下面是一个示例代码,展示了如何在应用程序的配置类中添加

@EnableTransactionManagement

注解:

import org.springframework.context.annotation.Configuration;
import org.springframework.transaction.annotation.EnableTransactionManagement;
@Configuration
@EnableTransactionManagement
public class AppConfig {
    // 配置其他的Bean
}

通过添加

@EnableTransactionManagement

注解,我们告诉Spring框架启用事务管理器,并确保在需要的时候正确创建和管理事务。这样就可以解决这个问题了。

总结起来,当遇到(javax.persistence.TransactionRequiredException: No EntityManager with actual transaction available for current thread)这个问题时,我们需要检查是否正确配置了事务管理器,并在应用程序的配置类上添加

@EnableTransactionManagement

注解来解决这个问题。

0
0 Comments

这个问题的出现的原因是在配置文件中缺少对ViewResolver bean的配置。可能是因为在问题中展示的root-context.xml文件中没有对ViewResolver bean进行配置。而在web.xml文件中可能存在用于Dispatcher Servlet(例如dispatcher-servlet.xml)的servlet context配置文件。请在dispatcher-servlet.xml中添加以下配置:

<tx:annotation-driven />

如果在dispatcher-servlet.xml中没有像其他xmlns:定义一样定义xmlns:tx,请添加该定义。请确保在dispatcher servlet context中有适当的component-scan配置。

<context:component-scan base-package="by.netcracker.artemyev.*" />

您还可以参考 doesn't work in Spring Web MVC?中报告了类似问题的帖子。

解决这个问题的方法是在dispatcher-servlet.xml中添加tx:annotation-driven的配置,并确保在dispatcher servlet context中有适当的component-scan配置。还可以参考相关帖子来解决类似的问题。

0