@Transactional踩坑实践,你能看的出来么?

前阵子接手了一段同事之前的代码,里面用到了@transaction注解,了解Spring的小伙伴肯定知道,@Transactional是Spring提供的一种控制事务管理的快捷手段。但是我这段程序在运行的时候,经常出现莫名其妙的问题,连夜研究了好久才搞清楚,在这里记录一下, 避免大家入坑。

1. 大家来找茬

在介绍具体问题之前,我把问题代码简化了一下,看大家能找到其中的问题吗?

问题代码1

下面的这段代码主要是想利用MySQL里面的行锁select for update,来实现简单的分布式锁。但是在实践过程中,发现这个锁好像并没有生效,而且在数据库的里面也没有查找对应transaction连接的信息。

@Component
@EnableScheduling
public class someService {
  
  @Scheduled(...)
  public doSomeWork() {
    // find some id by logic
    
    // process the related info
    doOtherWork(id);
  }
  
  @Transactional(isolation = Isolation.READ_COMMITTED)
  public void doOtherWork(id) {
    Info info = requestMapper.selectByPrimaryKeyForUpdate(id);
    doSomeFollowingProcess(info);
    ...
  }
}

问题代码2

下面代码分两个步骤,第一步会检查相关信息,第二步调用了一个transactional修饰的方法,完成一些基本工作;但在实践中,发现一个非常诡异的问题,在MainWork中,doSomeCheck执行时会抛出nullPointException,debug发现所有autowired进来的service均为空,注释掉doSomeCheck里面的内容后,继续往下执行,却发现doWork能够正常执行,所有的注入均没有问题。

@Component
public class MainWork {
  @AutoWired
  DetailWork detailWork
    
  public void workflow() {
    detailWork.doSomeCheck();
    detailWork.doWork();
  }
}
​
@Component
public class DetailWork {
  
    @AutoWired
    UsefulService usefulService;
  
    @AutoWired
    InfoService infoService;
  
    @Transactional(isolation = Isolation.READ_COMMITTED)
    public void doWork() {
      usefulService.doSomeWork();
    }
  
    void doSomeCheck() {
      infoService.getInfo();
    }
}

大伙看看能发现什么问题吗?

2.

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
@Transactional 是 Spring 框架中用于管理事务的注解。在使用 @Transactional 注解时,需要将该注解放置在需要进行事务管理的方法或类上。 在实际工程中,一般建议将 @Transactional 注解放置在 Service 层的方法上,因为 Service 层是业务逻辑的处理层,通常需要对多个 DAO 层的操作进行事务管理。 例如: ```java @Service public class UserService { @Autowired private UserDao userDao; @Transactional public void addUser(User user) { userDao.addUser(user); } @Transactional public void updateUser(User user) { userDao.updateUser(user); } @Transactional public void deleteUser(int userId) { userDao.deleteUser(userId); } } ``` 在上述代码中,UserService 类中的 addUser、updateUser 和 deleteUser 方法都加上了 @Transactional 注解,表示这些方法需要进行事务管理。当执行这些方法时,如果其中任意一个方法出现异常,Spring 框架会自动回滚该方法所执行的所有操作。 当然,在使用 @Transactional 注解时,还需要配置事务管理器。在 Spring Boot 中,可以通过在 application.properties 文件中配置相关的属性来实现事务管理。 例如: ```properties # 数据源配置 spring.datasource.driver-class-name=com.mysql.jdbc.Driver spring.datasource.url=jdbc:mysql://localhost:3306/test spring.datasource.username=root spring.datasource.password=root # 事务管理器配置 spring.jpa.properties.hibernate.current_session_context_class=org.springframework.orm.hibernate5.SpringSessionContext spring.jpa.show-sql=true spring.jpa.hibernate.ddl-auto=update spring.jpa.properties.hibernate.jdbc.batch_size=20 spring.jpa.properties.hibernate.order_inserts=true spring.jpa.properties.hibernate.order_updates=true ``` 总体来说,在实际工程中,使用 @Transactional 注解来管理事务是一种比较成熟的做法。但是需要注意的是,在使用 @Transactional 注解时,需要仔细分析业务逻辑,避免出现事务的嵌套和事务的传播问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值