Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/spring/11.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181

Warning: file_get_contents(/data/phpspider/zhask/data//catemap/5/spring-mvc/2.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Spring @事务性注释不起作用。未创建数据库行_Spring_Spring Mvc_Jpa_Transactional - Fatal编程技术网

Spring @事务性注释不起作用。未创建数据库行

Spring @事务性注释不起作用。未创建数据库行,spring,spring-mvc,jpa,transactional,Spring,Spring Mvc,Jpa,Transactional,下面是我所有的代码。未创建数据库行。没有引发异常 package com.rishi.app.models; import java.util.Collection; import javax.management.Query; import javax.persistence.Entity; import javax.persistence.EntityManager; import javax.persistence.GeneratedValue; import javax.persis

下面是我所有的代码。未创建数据库行。没有引发异常

package com.rishi.app.models;

import java.util.Collection;

import javax.management.Query;
import javax.persistence.Entity;
import javax.persistence.EntityManager;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.PersistenceContext;
import javax.persistence.PersistenceUnit;
import javax.persistence.TypedQuery;

@Entity
public class Customer {

    @Id
    @GeneratedValue(strategy=GenerationType.AUTO)
    private int id;
    private String firstName;
    private String lastName;

    protected Customer() {}

    public Customer(String firstName, String lastName) {
        this.firstName = firstName;
        this.lastName = lastName;
    }

    @Override
    public String toString() {
        return String.format(
                "Customer[id=%d, firstName='%s', lastName='%s']",
                id, firstName, lastName);
    }

}


package com.rishi.app.repositories;

import javax.persistence.EntityManager;
import javax.persistence.PersistenceContext;
import javax.transaction.Transactional;

import org.springframework.stereotype.Repository;

import com.rishi.app.models.Customer;

@Repository
public class CustomerRepository {
    @PersistenceContext
    private EntityManager em;

    @Transactional
    public void save(Customer c) {
        em.persist(c);
    }
}

package com.rishi.app.controllers;

import java.text.DateFormat;
import java.util.Date;
import java.util.Locale;

import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.PersistenceContext;
import javax.transaction.Transactional;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;

import com.rishi.app.models.Customer;
import com.rishi.app.repositories.CustomerRepository;

/**
 * Handles requests for the application home page.
 */
@Controller
public class HomeController {
    @PersistenceContext
    EntityManager entityManager;


    @Autowired
    EntityManagerFactory entityManagerFactory;

    @Autowired
    CustomerRepository customerRepository;

    private static final Logger logger = LoggerFactory.getLogger(HomeController.class);

    /**
     * Simply selects the home view to render by returning its name.
     * @throws Exception 
     */
    @Transactional
    @RequestMapping(value = "/", method = RequestMethod.GET)
    public String home(Locale locale, Model model) {
        logger.info("Welcome home controller! The client locale is {}.", locale);

        Customer c = new Customer("Rishi", "Paranjape");
        customerRepository.save(c);

            //Following 4 lines work just fine. Database row is actually created.
        //EntityManager em = entityManagerFactory.createEntityManager();
        //em.getTransaction().begin();
        //em.persist(c);
        //em.getTransaction().commit();

        Date date = new Date();
        DateFormat dateFormat = DateFormat.getDateTimeInstance(DateFormat.LONG, DateFormat.LONG, locale);

        String formattedDate = dateFormat.format(date);

        model.addAttribute("serverTime", formattedDate );
        //throw new Exception("bad stuff");
        return "home";
    }

}
我的servlet上下文xml:

<?xml version="1.0" encoding="UTF-8"?>
<beans:beans xmlns="http://www.springframework.org/schema/mvc"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:beans="http://www.springframework.org/schema/beans"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.1.xsd
        http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.1.xsd
        http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.1.xsd">

    <!-- DispatcherServlet Context: defines this servlet's request-processing infrastructure -->

    <!-- Enables the Spring MVC @Controller programming model -->
    <annotation-driven />

    <!-- Handles HTTP GET requests for /resources/** by efficiently serving up static resources in the ${webappRoot}/resources directory -->
    <resources mapping="/resources/**" location="/resources/" />

    <!-- Resolves views selected for rendering by @Controllers to .jsp resources in the /WEB-INF/views directory -->
    <beans:bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <beans:property name="prefix" value="/WEB-INF/views/" />
        <beans:property name="suffix" value=".jsp" />
    </beans:bean>

    <context:component-scan base-package="com.rishi.app" />


    <beans:bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
      <beans:property name="dataSource" ref="dataSource" />
      <beans:property name="packagesToScan" value="com.rishi.app.models" />
      <beans:property name="jpaVendorAdapter">
         <beans:bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter" />
      </beans:property>
      <beans:property name="jpaProperties">
         <beans:props>
            <beans:prop key="hibernate.hbm2ddl.auto">validate</beans:prop>
            <beans:prop key="hibernate.dialect">org.hibernate.dialect.MySQL5Dialect</beans:prop>
         </beans:props>
      </beans:property>
   </beans:bean>

   <beans:bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
      <beans:property name="driverClassName" value="com.mysql.jdbc.Driver" />
      <beans:property name="url" value="jdbc:mysql://localhost:3306/spring" />
      <beans:property name="username" value="rishi" />
      <beans:property name="password" value="password" />
   </beans:bean>

    <beans:bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
      <beans:property name="entityManagerFactory" ref="entityManagerFactory" />
   </beans:bean>
   <tx:annotation-driven mode="aspectj" transaction-manager="transactionManager" />


</beans:beans>

我有根上下文和servlet上下文。我的所有bean都在servlet上下文中,我的根上下文几乎为空。我的主要问题是em.persist被调用,但什么也不做。如果我调试应用程序,我可以看到正在调用CustomerRepository中的save方法。没有抛出异常或显示任何错误。但是,没有创建数据库行。

可能的原因是@Transactional应用于错误spring上下文中的bean

许多SpringMVC应用程序有两个上下文:一个根上下文通常用于事务性服务/存储库等公共bean,另一个特定于web的上下文包含控制器,有关更多详细信息,请参见此

似乎正在发生的事情是,tx:annotation-driven正在应用于既不存在控制器也不存在存储库的上下文

看起来@Transactional应用于根上下文,并将所有bean放在dispatcher上下文中

如果是这种情况,请将tx:annotation-driven移动到定义bean的XML文件或相应的组件扫描。这将在bean确实存在的上下文中应用@Transactional

通常,作为一般最佳实践,我们不在控制器上应用@Transactional,也不在存储库上应用@Transactional,而是在中间服务层bean中应用@service,其中包含业务逻辑


但是存储库和控制器只是Springbean,所以如果您想使用@transactional,那么它也可以工作。

可能的原因是@transactional应用于错误spring上下文中的bean

许多SpringMVC应用程序有两个上下文:一个根上下文通常用于事务性服务/存储库等公共bean,另一个特定于web的上下文包含控制器,有关更多详细信息,请参见此

似乎正在发生的事情是,tx:annotation-driven正在应用于既不存在控制器也不存在存储库的上下文

看起来@Transactional应用于根上下文,并将所有bean放在dispatcher上下文中

如果是这种情况,请将tx:annotation-driven移动到定义bean的XML文件或相应的组件扫描。这将在bean确实存在的上下文中应用@Transactional

通常,作为一般最佳实践,我们不在控制器上应用@Transactional,也不在存储库上应用@Transactional,而是在中间服务层bean中应用@service,其中包含业务逻辑


但是存储库和控制器只是Springbean,所以如果您想使用@transactional,也可以使用它。

首先为您的存储库编写一个集成测试。这将更容易缩小问题是在控制器中还是在存储库中

从理论上讲,@Transactional repository save方法不在接口中,因此不会为其创建JDK动态代理,除非将proxy target class=true添加到元素中,并将CGLIB添加到类路径中。如果未创建代理,则没有执行事务性工作的建议。您可以通过在存储库中设置断点并查看堆栈帧来检查是否提到TransactionInterceptor类

实际上,如果缺少事务代理,我希望得到一个异常,即您至少在Hibernate中没有打开的会话。由于没有出现异常,我怀疑Spring正在装饰控制器的@Transactional,这不是事务问题


要测试它是否是事务性问题,请在保存后调用flush,在flush之后但在事务关闭之前放置一个断点,并在其他地方进行集成测试、DB console,以创建具有READ_UNCOMMITTED隔离的新事务,然后检查您是否可以看到该行。

首先为您的存储库编写一个集成测试。这将更容易缩小问题是在控制器中还是在存储库中

从理论上讲,@Transactional repository save方法不在接口中,因此不会为其创建JDK动态代理,除非将proxy target class=true添加到元素中,并将CGLIB添加到类路径中。如果未创建代理,则没有执行事务性工作的建议。您可以通过在存储库中设置断点并查看堆栈帧来检查是否提到TransactionInterceptor类

实际上,如果缺少事务代理,我希望得到一个异常,即您至少在Hibernate中没有打开的会话。由于没有出现异常,我怀疑Spring正在装饰控制器的@Transactional,而这不是tran 销售问题

要测试它是否是事务性问题,请在保存后调用flush,在flush之后但事务关闭之前放置一个断点,并在其他地方集成测试DB console,创建具有读取未提交隔离的新事务,并检查是否可以脏读查看该行。

删除模式=aspectj,因为这在没有JVM代理的情况下不起作用,请参见

删除它将允许Spring使用非aspectj编织机制JDK代理或CGLIB代理(如果适用),因此@Transactional应该在servlet上下文中工作任何bean。

Remove mode=aspectj,因为如果没有JVM代理,这将无法工作,请参见此



删除此选项将允许Spring使用非aspectj编织机制JDK代理或CGLIB代理(如果适用),因此@Transactional应该在servlet上下文中使用任何bean。

您使用的是javax.transaction.Transactional,虽然您应该真正使用org.springframework.transaction.annotation.Transactional。

但您使用的是javax.transaction.Transactional,虽然你应该真正使用org.springframework.transaction.annotation.Transactional.

你也可以发布你的web.xml吗?不错的类,但我想你也掉进了重复组件扫描的陷阱,这会导致CustomerRepository的代理和未代理实例,这基本上会使所有AOP(包括事务)都变得无用。在看到servlet上下文文件后,我在下面添加了第二个答案。您是否尝试删除mode=aspectj,结果如何?您是否也可以发布您的web.xml?不错的类,但我猜您也落入了重复组件扫描的陷阱,这会导致CustomerRepository的代理和未代理实例,这基本上会使所有AOP(包括事务)都变得无用。在看到servlet上下文文件后,我在下面添加了第二个答案。您是否尝试删除mode=aspectj,结果如何?我有根上下文和servlet上下文。我的所有bean都在servlet上下文中,我的根上下文几乎为空。我的主要问题是em.persist正在被调用。如果我调试应用程序,我可以看到正在调用CustomerRepository中的save方法。没有抛出异常或显示任何错误。但是,没有创建任何数据库行。如果tx:annotation-driven仅在根上下文中,这将解释为什么servlet上下文中的bean不是事务性的。尝试在servlet上下文中添加tx:annotation驱动。否则,如果您发布servlet上下文xml和根上下文xml,这将有助于确定解决方案。tx注释驱动在servlet上下文中,而不是根上下文中。上下文:组件扫描或XMLBean定义,是否也在servlet上下文中?因为如果它在根上下文中,@Transactional不起作用是正常的,请看这个答案@khadesdev:请查看更新的问题。很抱歉之前没有发布servlet上下文。我有根上下文和servlet上下文。我的所有bean都在servlet上下文中,我的根上下文几乎为空。我的主要问题是em.persist正在被调用。如果我调试应用程序,我可以看到正在调用CustomerRepository中的save方法。没有抛出异常或显示任何错误。但是,没有创建任何数据库行。如果tx:annotation-driven仅在根上下文中,这将解释为什么servlet上下文中的bean不是事务性的。尝试在servlet上下文中添加tx:annotation驱动。否则,如果您发布servlet上下文xml和根上下文xml,这将有助于确定解决方案。tx注释驱动在servlet上下文中,而不是根上下文中。上下文:组件扫描或XMLBean定义,是否也在servlet上下文中?因为如果它在根上下文中,@Transactional不起作用是正常的,请看这个答案@khadesdev:请查看更新的问题。很抱歉之前没有发布servlet上下文。这不完全正确,aspectj模式也适用于编译时编织。这不完全正确,aspectj模式也适用于编译时编织。