Spring - No EntityManager with actual transaction available for current thread - cannot reliably process 'persist' call

前端 未结 15 1533
自闭症患者
自闭症患者 2020-11-30 20:08

I get this error when trying to invoke \"persist\" method to save entity model to database in my Spring MVC web application. Can\'t really find any post or page in internet

相关标签:
15条回答
  • 2020-11-30 20:11

    I had the same error when accessing an already transactional-annotated method from a non-transactional method within the same component:

    Before:
        @Component
        public class MarketObserver {
            @PersistenceContext(unitName = "maindb")
            private EntityManager em;
    
            @Transactional(value = "txMain", propagation = Propagation.REQUIRES_NEW)
            public void executeQuery() {
              em.persist(....);
            }
    
    
            @Async
            public void startObserving() {
              executeQuery(); //<-- Wrong
            }
        }
    
        //In another bean:
         marketObserver.startObserving();
    

    I fixed the error by calling the executeQuery() on the self-referenced component:

    Fixed version:
        @Component
        public class MarketObserver {
            @PersistenceContext(unitName = "maindb")
            private EntityManager em;
    
            @Autowired
            private GenericApplicationContext context;
    
            @Transactional(value = "txMain", propagation = Propagation.REQUIRES_NEW)
            public void executeQuery() {
              em.persist(....);
            }
    
    
            @Async
            public void startObserving() {
              context.getBean(MarketObserver.class).executeQuery(); //<-- Works
            }
        }
    
    0 讨论(0)
  • 2020-11-30 20:16

    Just a note for other users searching for answers for thie error. Another common issue is:

    You generally cannot call an @transactional method from within the same class.

    (There are ways and means using AspectJ but refactoring will be way easier)

    So you'll need a calling class and class that holds the @transactional methods.

    0 讨论(0)
  • 2020-11-30 20:20

    Adding the org.springframework.transaction.annotation.Transactional annotation at the class level for the test class fixed the issue for me.

    0 讨论(0)
  • 2020-11-30 20:21

    This error had me foxed for three days, the situation I faced produced the same error. Following all the advice I could find, I played with the configuration but to no avail.

    Eventually I found it, the difference, the Service I was executing was contained in a common jar, the issue turned out to be AspectJ not treating the Service instantiation the same. In effect the proxy was simply calling the underlying method without all the normal Spring magic being executed before the method call.

    In the end the @Scope annotation placed on the service as per the example solved the issue:

    @Service
    @Scope(proxyMode = ScopedProxyMode.INTERFACES)
    @Transactional
    public class CoreServiceImpl implements CoreService {
        @PersistenceContext
        protected EntityManager entityManager;
    
        @Override
        public final <T extends AbstractEntity> int deleteAll(Class<T> clazz) {
            CriteriaDelete<T> criteriaDelete = entityManager.getCriteriaBuilder().createCriteriaDelete(clazz);
            criteriaDelete.from(clazz);
            return entityManager.createQuery(criteriaDelete).executeUpdate();
        }
    
    }
    

    The method I have posted is a delete method but the annotations affect all persistence methods in the same way.

    I hope this post helps someone else who has struggled with the same issue when loading a service from a jar

    0 讨论(0)
  • 2020-11-30 20:21

    I had the same error because I switched from XML- to java-configuration.

    The point was, I didn't migrate <tx:annotation-driven/> tag, as Stone Feng suggested.

    So I just added @EnableTransactionManagement as suggested here Setting Up Annotation Driven Transactions in Spring in @Configuration Class, and it works now

    0 讨论(0)
  • 2020-11-30 20:23

    For us, the problem came down to same context settings in multiple configuration files. Check you've not duplicated the following in multiple config files.

    <context:property-placeholder location="classpath*:/module.properties"/>
    <context:component-scan base-package="...." />
    
    0 讨论(0)
提交回复
热议问题