【问题标题】:@Transactional is not starting a transaction in standalone application@Transactional 没有在独立应用程序中启动事务
【发布时间】:2017-04-12 13:01:57
【问题描述】:

我创建了以下示例。但我没有得到预期的结果。 这是我的代码:

1.application.properties

################### JDBC Configuration ##########################
jdbc.driverClassName=org.hsqldb.jdbcDriver
jdbc.url=jdbc:hsqldb:file:db/hsqldb;shutdown=true
jdbc.username=root
jdbc.password=


################### Hibernate Configuration ##########################
hibernate.dialect=org.hibernate.dialect.HSQLDialect
hibernate.show_sql=true
hibernate.hbm2ddl.auto=update
hibernate.generate_statistics=true

2。数据库配置.java

/**
 * 
 * load Hibernate Session factory to get session ,load Hibernate Transaction
 * Manager to deal with transaction
 * 
 */

@Configuration
@EnableTransactionManagement
public class DatabaseConfig {

    @Value("${jdbc.driverClassName}")
    private String driverClass;
    @Value("${jdbc.url}")
    private String url;
    @Value("${jdbc.username}")
    private String username;
    @Value("${jdbc.password}")
    private String password;
    @Value("${hibernate.dialect}")
    private String dialect;
    @Value("${hibernate.show_sql}")
    private String showSQL;
    @Value("${hibernate.hbm2ddl.auto}")
    private String hbm2ddl;

    @Bean
    public Properties hibernateProperties() {

        Properties hp = new Properties();
        hp.put("hibernate.dialect", dialect);
        hp.put("hibernate.show_sql", showSQL);
        hp.put("hibernate.hbm2ddl.auto", hbm2ddl);
        hp.put("hibernate.current_session_context_class", "thread");
        return hp;

    }

    @Bean
    public DataSource driverManagerDataSource() {
        DriverManagerDataSource ds = new DriverManagerDataSource();
        ds.setDriverClassName(driverClass);
        ds.setUrl(url);
        ds.setUsername(username);
        ds.setPassword(password);
        return ds;
    }

    @Bean
    public LocalSessionFactoryBean sessionFactory() {
        LocalSessionFactoryBean sessionFactoryBean = new LocalSessionFactoryBean();
        sessionFactoryBean.setDataSource(driverManagerDataSource());
        sessionFactoryBean.setHibernateProperties(hibernateProperties());
        sessionFactoryBean.setPackagesToScan(new String[] { "com.orbit.hibernate.entities" });

        return sessionFactoryBean;
    }

    @Bean
    @Autowired
    public HibernateTransactionManager transactionManager(SessionFactory sessionFactory) {
        HibernateTransactionManager tm = new HibernateTransactionManager(sessionFactory);
        return tm;

    }

}

3. AppConfig.java

/**
 * 
 * This class has all the beans required at application level
 * 
 */
@Configuration
@Import(DatabaseConfig.class)
@ComponentScan("com.orbit")
public class AppConfig {

    @Bean
    public static PropertyPlaceholderConfigurer propertyPlaceHolderConfigurer() {
        PropertyPlaceholderConfigurer pphc = new PropertyPlaceholderConfigurer();
        pphc.setLocation(new ClassPathResource("application.properties"));
        pphc.setIgnoreUnresolvablePlaceholders(true);
        return pphc;
    }
}

4.User.java

@Entity
public class User {

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private long id;

    private String name;

    public long getId() {
        return id;
    }

    public void setId(long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

5. UserRepository.java

@Repository
public class UserRepository {

    @Autowired
    private SessionFactory sessionFactory;

    public void save() {
        User u = new User();
        String random = UUID.randomUUID().toString();
        u.setName("mks" + random);
        Session session = sessionFactory.getCurrentSession();
        session.persist(u);

    }

    @SuppressWarnings("unchecked")
    public List<User> getAll() {
        Session session = sessionFactory.getCurrentSession();
        Query q = session.createQuery("FROM User");
        List<User> users = q.getResultList();
        return users;
    }
}

6.用户服务.java

@Service
@Transactional
public class UserService {

    @Autowired
    private UserRepository userRepo;

    public List<User> getAllUser() {
        return userRepo.getAll();
    }

    public void save() {
        userRepo.save();
    }
}

7. AppLaunch.java

public class LaunchApp {

    public static void main(String[] args) {
        AnnotationConfigApplicationContext ctx = new AnnotationConfigApplicationContext(AppConfig.class);
        UserService userService = ctx.getBean(UserService.class);
        userService.save();
        userService.save();
        System.out.println("Printing all user");
        List<User> user = userService.getAllUser();
        user.forEach(System.out::println);
        ctx.close();
    }
}

当我运行 AppLaunch.java 类时。发生以下执行:

线程 "main" org.hibernate.HibernateException 中的异常:如果没有活动事务,持久化无效

这是我的 pom.xml:

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.orbit</groupId>
    <artifactId>spring.hibernate.java.config</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <packaging>jar</packaging>

    <name>spring.hibernate.java.config</name>
    <url>http://maven.apache.org</url>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    </properties>

    <dependencies>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>3.8.1</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context-support</artifactId>
            <version>4.3.7.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-jdbc</artifactId>
            <version>4.3.7.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-orm</artifactId>
            <version>4.3.7.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-tx</artifactId>
            <version>4.3.7.RELEASE</version>
        </dependency>

        <dependency>
            <groupId>org.hibernate</groupId>
            <artifactId>hibernate-core</artifactId>
            <version>5.2.9.Final</version>
        </dependency>
        <dependency>
            <groupId>org.hsqldb</groupId>
            <artifactId>hsqldb</artifactId>
            <version>2.3.4</version>
        </dependency>
    </dependencies>
</project>

我做错了什么,在这里?

这是完整的堆栈跟踪:

2017 年 4 月 12 日晚上 8:16:35 org.springframework.context.annotation.AnnotationConfigApplicationContext prepareRefresh INFO:正在刷新 org.springframework.context.annotation.AnnotationConfigApplicationContext@fa5d3: 启动日期 [Wed Apr 12 20:16:35 IST 2017];上下文层次的根 2017 年 4 月 12 日晚上 8:16:36 org.springframework.jdbc.datasource.DriverManagerDataSource setDriverClassName INFO:加载的 JDBC 驱动程序:org.hsqldb.jdbcDriver Apr 2017 年 12 月 12 日晚上 8:16:36 org.hibernate.Version logVersion 信息:HHH000412: Hibernate Core {5.2.9.Final} 2017 年 4 月 12 日晚上 8:16:36 org.hibernate.cfg.Environment 信息:HHH000206: hibernate.properties 未找到 2017 年 4 月 12 日晚上 8:16:37 org.hibernate.annotations.common.reflection.java.JavaReflectionManager 信息:HCANN000001:Hibernate Commons 注释 {5.0.1.Final} 线程“main”中的异常 org.hibernate.HibernateException:没有活动的持久化无效 交易于 org.hibernate.context.internal.ThreadLocalSessionContext$TransactionProtectionWrapper.invoke(ThreadLocalSessionContext.java:351) 在 com.sun.proxy.$Proxy27.persist(未知来源) com.orbit.repositories.UserRepository.save(UserRepository.java:26) 在 com.orbit.services.UserService.save(UserService.java:25) 在 com.orbit.services.UserService$$FastClassBySpringCGLIB$$6492a26b.invoke() 在 org.springframework.cglib.proxy.MethodProxy.invoke(MethodProxy.java:204) 在 org.springframework.aop.framework.CglibAopProxy$CglibMethodInvocation.invokeJoinpoint(CglibAopProxy.java:721) 在 org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:157) 在 org.springframework.transaction.interceptor.TransactionInterceptor$1.proceedWithInvocation(TransactionInterceptor.java:99) 在 org.springframework.transaction.interceptor.TransactionAspectSupport.invokeWithinTransaction(TransactionAspectSupport.java:282) 在 org.springframework.transaction.interceptor.TransactionInterceptor.invoke(TransactionInterceptor.java:96) 在 org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179) 在 org.springframework.aop.framework.CglibAopProxy$DynamicAdvisedInterceptor.intercept(CglibAopProxy.java:656) 在 com.orbit.services.UserService$$EnhancerBySpringCGLIB$$15a51388.save() 在 com.orbit.main.LaunchApp.main(LaunchApp.java:16)

【问题讨论】:

  • 在您的代码中未启动任何事务,可能将propagation = Propagation.REQUIRES_NEW 添加到您的@Transactional 将解决您的问题
  • @Nemesis:我也厌倦了。结果还是一样
  • 尝试在注释中指定事务管理器,如下所示:@Transactional(value="transactionManager")
  • 我也试过这个,给 @Bean("txManager") 然后在 @Transcational 中指定 transactionManager 属性
  • transactionmanager 未在您的@configuration 中配置。您可能希望对其进行配置以使其正常工作。 docs.spring.io/spring-framework/docs/4.2.x/…

标签: java hibernate spring-jdbc spring-transactions spring-java-config


【解决方案1】:

尝试删除此属性并再次运行。

hp.put("hibernate.current_session_context_class", "thread");

【讨论】:

  • 我正在使用上下文会话。因此它是必需的。
  • 尝试使用这个注解@EnableTransactionManagement
  • 您需要删除该属性,因为该属性会破坏弹簧的正确集成。 Spring 有自己的 hibernate 上下文会话实现。
  • @M.Deinum :我怎样才能获得上下文会话详细信息
  • 哪些细节?你不需要做什么?不确定您对该评论的要求。
【解决方案2】:

需要指定spring事务文档https://docs.spring.io/spring/docs/current/spring-framework-reference/htmlsingle/#testing-tx中指定的事务管理器

另外请编辑 save 方法并添加 beginTransaction 如下:

    public void save() {
        User u = new User();
        String random = UUID.randomUUID().toString();
        u.setName("mks" + random);
        Session session = getSessionFactory().getCurrentSession();
        Transaction trans = session.beginTransaction();
        session.persist(u);
        trans.commit();

    }

【讨论】:

  • 相关吗?
  • 您将在此处真正了解使用@Transactional 的来龙去脉
  • Kayv:我知道你在链接中提到的内容。我的问题是我没有运行任何测试。那里的相关性也是如此。我实际上想在独立应用程序中使用基于 Spring 应用程序的事务功能
  • 它会起作用,但问题是为什么上面的代码不起作用,我是不是对 Spring 事务管理理解不正确,或者缺少什么?
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2014-11-05
  • 2016-03-16
  • 2012-04-10
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多