做练习的时候出现了一个异常:
javax.persistence.TransactionRequiredException: No EntityManager with actual transaction available for current thread - cannot reliably process 'remove' call
org.springframework.orm.jpa.SharedEntityManagerCreator$SharedEntityManagerInvocationHandler.invoke(SharedEntityManagerCreator.java:289)
com.sun.proxy.$Proxy37.remove(Unknown Source)
pers.kevin.mvc.rest.plus.plus.dao.impl.BookDaoImpl.deleteBook(BookDaoImpl.java:37)
pers.kevin.mvc.rest.plus.plus.service.impl.BookServiceImpl.deleteBook(BookServiceImpl.java:31)
pers.kevin.mvc.rest.plus.plus.controller.BookController.deleteBook(BookController.java:43)
sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
java.lang.reflect.Method.invoke(Method.java:498)
org.springframework.web.method.support.InvocableHandlerMethod.doInvoke(InvocableHandlerMethod.java:209)
org.springframework.web.method.support.InvocableHandlerMethod.invokeForRequest(InvocableHandlerMethod.java:136)
org.springframework.web.servlet.mvc.method.annotation.ServletInvocableHandlerMethod.invokeAndHandle(ServletInvocableHandlerMethod.java:102)
org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.invokeHandlerMethod(RequestMappingHandlerAdapter.java:877)
org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.handleInternal(RequestMappingHandlerAdapter.java:783)
org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter.handle(AbstractHandlerMethodAdapter.java:87)
org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:991)
org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:925)
org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:974)
org.springframework.web.servlet.FrameworkServlet.doDelete(FrameworkServlet.java:899)
javax.servlet.http.HttpServlet.service(HttpServlet.java:667)
org.springframework.web.servlet.FrameworkServlet.service(FrameworkServlet.java:851)
javax.servlet.http.HttpServlet.service(HttpServlet.java:742)
org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:52)
org.springframework.web.filter.CharacterEncodingFilter.doFilterInternal(CharacterEncodingFilter.java:200)
org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107)
org.springframework.web.filter.HiddenHttpMethodFilter.doFilterInternal(HiddenHttpMethodFilter.java:93)
org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107)
org.springframework.web.filter.HttpPutFormContentFilter.doFilterInternal(HttpPutFormContentFilter.java:109)
org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107)
从错误上看应该是事务的问题,但是我找了好久,排除了事务配置的问题,最后确定是可能事务没有注入进来,导致没有事务。
通过Spring和SpringMVC的配置查看,发现了问题:
在SpringMVC中配置扫描包如下:
<context:component-scan base-package="pers.kevin.mvc.rest"/>
然后Spring中配置的扫描包如下:
<context:component-scan base-package="pers.kevin.mvc.rest"></context:component-scan>
根本原因:子容器扫描装配了@Service 注解的实例。
两个容器扫描都是根路径,这就会引发Spring和SpringMVC父子容器的问题。SpringMVC也会扫描到service层,并且在子容器中实例化了service,而当需要注入service的时候,SpringMVC会就近找子容器的东西,找不到才去父容器Spring中找,当Controller要注入Service的时候,首先找到了SpringMVC中实例好的Service,而该Service实例理应由父容器Spring进行初始化以保证事务的增强处理,所以在子容器SpringMVC的Service没有经过事务加强处理,故而没有事务处理能力。。
所以解决方法就是在SpringMVC配置的扫描包中只需要扫描到Controller。。
<context:component-scan base-package="pers.kevin.mvc.rest.controller"/>
关于Spring和SpringMVC父子容器问题的参考链接: