java

关注公众号 jb51net

关闭
首页 > 软件编程 > java > @Scheduled定时器使用@Thransactional问题

解决@Scheduled定时器使用@Thransactional事物问题

作者:墨止戈

这篇文章主要介绍了解决@Scheduled定时器使用@Thransactional事物问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

@Scheduled定时器使用@Thransactional问题

最近遇到一个非常棘手的问题

当jpa使用update/delete时如果不添加@Thransactional会报

Executing an update/delete query;

nested exception is javax.persistence.TransactionRequiredException: Executing anupdate/delete query异常

可是我添加@Thransactional还是会报这个异常

问了好多同行,他们都是告诉我定时器不能使用事物,实际并不是这样的,我查看了很多资料是可以使用的,@Scheduled和@Thransaction不能在同一个类中。

并且实体类要加上@EnableTransactionManagement注解,使我们这个工程支持事物

后来我试了很多

@Thransactional(rolbackFor=ThrowAble.class)
@Thransactional(rolbackFor=Exception.class)
@Thransactional(ReadOnly=false)//jpa默认不开启事物 等等就是想使事物进行回滚 可是造化弄人呀

废话不多说,直接上最关键的!!

// 在连接数据库也就是date jpa 工程的启动类中 配置
@Bean(name="transactionManager")
public PlatformTransactuibManager configurationTm(EntityManagerFactory factory){
	return new JpaTransactionManager(factory);
}

关于Scheduled定时任务

认识定时器

https://cron.qqe2.com/

定时器表达式生成器

定时器场景使用

/*
    使用定时任务关闭超期未支付订单,会存在的弊端
        1、会有时间差,导致程序不严谨
        2、不支持集群:单机没毛病,使用集群后,就会有多个定时任务
            解决方案:只使用一台计算机节点,单独用来运行所有的定时任务
        3、会全表检索数据库,对数据库的性能有极大的影响:数据量大的情况
    定时任务,仅仅只适用于小型轻量级项目,传统项目

    最好使用消息队列 :MQ:RabbitMQ、RockerMQ、Kafka、ZeroMQ...
        延时任务(队列)
 */

定时器的使用

1、需要定时执行的方法上加上@Scheduled注解,这个注解中可以指定定时执行的规则,稍后详细介绍。

2、Spring容器中使用@EnableScheduling开启定时任务的执行,此时spring容器才可以识别@Scheduled标注的方法,然后自动定时执行。

Component

package com.laity.config;

import com.laity.service.OrderService;
import com.laity.utils.DateUtil;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;

/**
 * @author: Laity
 * @Project: JavaLaity
 * @Package: com.laity.config.OrderJob
 * @Date: 2022年08月12日 11:55
 * @Description: 定时任务
 * <p>
 * https://cron.qqe2.com/ cron表达式网站
 */
@Component
public class OrderJob {


    @Autowired
    private OrderService orderService;

    /*
        使用定时任务关闭超期未支付订单,会存在的弊端
            1、会有时间差,导致程序不严谨
            2、不支持集群:单机没毛病,使用集群后,就会有多个定时任务
                解决方案:只使用一台计算机节点,单独用来运行所有的定时任务
            3、会全表检索数据库,对数据库的性能有极大的影响:数据量大的情况
        定时任务,仅仅只适用于小型轻量级项目,传统项目

        最好使用消息队列 :MQ:RabbitMQ、RockerMQ、Kafka、ZeroMQ...
            延时任务(队列)
     */

    /*定时检索未支付的订单,并将其关闭*/
    //@Scheduled(cron = "0/3 * * * * ?")
    @Scheduled(cron = "* * 0/1 * * ?")
    public void autoCloseOrder() {
        orderService.closeOrder();
//        System.out.println("执行定时任务,当前时间为:" + DateUtil.getCurrentDateString(DateUtil.DATETIME_PATTERN));
    }
}

启动类

在启动类中开启定时器

package com.laity;

import lombok.extern.slf4j.Slf4j;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.web.servlet.ServletComponentScan;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.scheduling.annotation.EnableScheduling;
import org.springframework.transaction.annotation.EnableTransactionManagement;
import tk.mybatis.spring.annotation.MapperScan;

/**
 * @author: Laity
 * @Project: JavaLaity
 * @Package: com.laity.Application
 * @Date: 2022年07月19日 15:55
 * @Description: 自动装配注解
 * @EnableAutoConfiguration
 */
@Slf4j
@ServletComponentScan
@SpringBootApplication
//@EnableTransactionManagement /*开启事务管理 - 但是因为springboot自动装配(事务的自动装配)的关系我们不去使用这个注释*/
@MapperScan(basePackages = "com.laity.mapper")  /*扫描 mybatis 通用 mapper 所在的包*/
// 扫描所有包 以及 相关组件包 idworker => 主键生成
@ComponentScan(basePackages = {"com.laity", "org.n3r.idworker"})
@EnableScheduling  /*开启定时任务*/
public class Application {
    public static void main(String[] args) {
        ConfigurableApplicationContext run = SpringApplication.run(Application.class, args);
        log.info(run.toString());
    }
}

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

您可能感兴趣的文章:
阅读全文