【Java】catch里面抛出了异常finally里面的事务会提交吗?

文章目录

  • 背景
  • 目前的代码
  • 直接实战演示
  • 单元测试
  • 总结

背景

我们公司的系统中有一个业务场景,需要第三方的账户数据同步到我们系统。
同步账号的同时,会将所有同步数据和是否成功记录到一张同步日志表中,方便排查问题和记录。
好了,话不多说,我们直接上代码。

目前的代码

下面是一段伪代码

@Data
@Build
public class Test() {
 private boolean success = true;
}


@Transaction
public void sync() {
 Test test = Test.builder().build();
 try{
  xxxxx
 }catch(Exception e) {
   log.error("xxxx",e)
   test.setSuccess(false);
   throw new ThirdAccountException("同步账号错误:" + e.getMessage());
 } finally {
   testMapper.insert(test);
 }

}

大家能看出来这段代码有什么问题吗?

直接实战演示

数据库新建 账户数据同步记录表

CREATE TABLE `account_log` (
  `id` bigint NOT NULL,
  `data` varchar(255)  DEFAULT NULL COMMENT '第三方数据',
  `success` tinyint(1) DEFAULT NULL COMMENT '是否成功0否1是',
  `error_msg` varchar(255) DEFAULT NULL COMMENT '错误信息',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;

第三方账户数据实体

package com.fandf.demo.transaction;  
  
import cn.hutool.json.JSONUtil;  
import io.swagger.annotations.ApiModel;  
import lombok.AllArgsConstructor;  
import lombok.Data;  
  

@Data  
@AllArgsConstructor(staticName = "of")  
@ApiModel("第三方数据实体")  
public class ThirdAccount {  
  
    private String id;  
    private String data;  


    public AccountLog toAccountLog() {  
        return AccountLog.builder().data(JSONUtil.toJsonStr(this)).build();  
    }  
  
}

本地账户同步记录实体

package com.fandf.demo.transaction;  
  
import com.baomidou.mybatisplus.annotation.IdType;  
import com.baomidou.mybatisplus.annotation.TableId;  
import com.baomidou.mybatisplus.annotation.TableName;  
import com.baomidou.mybatisplus.extension.activerecord.Model;  
import io.swagger.annotations.ApiModelProperty;  
import lombok.Builder;  
import lombok.Data;  
import lombok.EqualsAndHashCode;  
  

@EqualsAndHashCode(callSuper = true)  
@TableName("account_log")  
@Data  
@Builder  
public class AccountLog extends Model<AccountLog> {  
  
    private static final long serialVersionUID = 5648238459610595434L;  

    @TableId(type = IdType.ASSIGN_ID)  
    private Long id;  

    @ApiModelProperty("第三方原始数据")  
    private String data;  

    @ApiModelProperty("是否成功: 0否1是")  
    private boolean success = true;  

    @ApiModelProperty("错误数据")  
    private String errorMsg;  
  
}

本地账户同步记录实体mapper

package com.fandf.demo.transaction;  
  
import com.baomidou.mybatisplus.core.mapper.BaseMapper;  
import org.springframework.stereotype.Repository;  
  

@Repository  
public interface AccountLogMapper extends BaseMapper<AccountLog> {  
}

同步账户处理的逻辑

package com.fandf.demo.transaction;  
  
import org.springframework.stereotype.Service;  
  
import javax.annotation.Resource;  
  

@Service  
public class TestTransaction {  
  
    @Resource  
    AccountLogMapper accountLogMapper;  
  
    @Transactional(rollbackFor = Exception.class)
    public void syncAccount(ThirdAccount account) {  
        AccountLog accountLog = account.toAccountLog();  
        try {  
            //模拟id为2 则抛出异常  
            if ("2".equals(account.getId())) {  
                throw new Exception("模拟抛出异常");  
            }  
        } catch (Exception e) {  
            accountLog.setSuccess(false);  
            accountLog.setErrorMsg(e.getMessage());  
            throw new IllegalArgumentException("同步第三方账号错误:" + e.getMessage());
        } finally {  
            accountLogMapper.insert(accountLog);  
        }  
    }  
  
  
}

单元测试

插入成功案例

package com.fandf.demo.transaction;  
  
import org.junit.jupiter.api.Test;  
import org.springframework.boot.test.context.SpringBootTest;  
  
import javax.annotation.Resource;  
  
import static org.junit.jupiter.api.Assertions.*;  
  
@SpringBootTest  
class TestTransactionTest {  
    @Resource  
    TestTransaction testTransaction;  
  
    @Test  
    void syncAccount() {  
        testTransaction.syncAccount(ThirdAccount.of("1", "成功数据"));  
    }  
}

查看数据库
【Java】catch里面抛出了异常finally里面的事务会提交吗?_第1张图片

是插入了,但是成功的success应该为1啊,为什么插入了0。
AccountLog.java

@ApiModelProperty("是否成功: 0否1是")  
    private boolean success = true;  

第三方转为AccountLog实体的方法

 public AccountLog toAccountLog() {  
        return AccountLog.builder().data(JSONUtil.toJsonStr(this)).build();  
    } 

我们来看看编译后的AccountLog.class源码中的AccountLogBuilder部分 success并未赋初始值

public static class AccountLogBuilder {  
    private Long id;  
    private String data;  
    private boolean success;  
    private String errorMsg;  

    AccountLogBuilder() {  
    }  

    public AccountLogBuilder id(Long id) {  
        this.id = id;  
        return this;  
    }  

    public AccountLogBuilder data(String data) {  
        this.data = data;  
        return this;  
    }  

    public AccountLogBuilder success(boolean success) {  
        this.success = success;  
        return this;  
    }  

    public AccountLogBuilder errorMsg(String errorMsg) {  
        this.errorMsg = errorMsg;  
        return this;  
    }  

    public AccountLog build() {  
        return new AccountLog(this.id, this.data, this.success, this.errorMsg);  
    }  

    public String toString() {  
        return "AccountLog.AccountLogBuilder(id=" + this.id + ", data=" + this.data + ", success=" + this.success + ", errorMsg=" + this.errorMsg + ")";  
    }  
}

我们看到Builder()方法返回了

AccountLogBuilder() {
}

对象。
指定data

public AccountLogBuilder data(String data) {  
    this.data = data;  
    return this;  
}

执行build()方法

public AccountLog build() {  
    return new AccountLog(this.id, this.data, this.success, this.errorMsg);  
}

success并未赋初始值,所以success=false, 存到数据库就是0了。
那么怎么解决呢?

  1. 字段加上注解@Builder.Default
@Builder.Default  
private boolean success = true;

我们再来测试下

@Test  
void syncAccount() {  
    testTransaction.syncAccount(ThirdAccount.of("1", "加上@Builder.Default成功数据"));  
}

查看数据库
【Java】catch里面抛出了异常finally里面的事务会提交吗?_第2张图片

插入成功。

  1. 手动赋值
 public AccountLog toAccountLog() {  
        return AccountLog.builder().success(true).data(JSONUtil.toJsonStr(this)).build();  
    } 

插入失败案例

@Test  
void syncAccount() {  
    testTransaction.syncAccount(ThirdAccount.of("2", "测试失败数据"));  
}

【Java】catch里面抛出了异常finally里面的事务会提交吗?_第3张图片

查看数据库
【Java】catch里面抛出了异常finally里面的事务会提交吗?_第4张图片

错误数据并没有插入进来。
这是因为catch里面抛出了异常,finally里面提交的事务也回滚了,我们去掉syncAccount(ThirdAccount account)方法上面的@Transactional注解。
再执行一次单元测试

@Test  
void syncAccount() {  
    testTransaction.syncAccount(ThirdAccount.of("2", "去掉@Transactional注解测试失败数据"));  
}

查看数据库
【Java】catch里面抛出了异常finally里面的事务会提交吗?_第5张图片

总结

  1. Build并不是对属性赋予默认值,如果想指定默认值可以在字段上使用@Builder.Default注解。
  2. 如果方法上加了@Transaction注解,catch里面抛出了异常,finally里面的事务会回滚。

当然我们在使用@Transaction注解的时候也需要注意事务的粒度,不能图省事直接在入口加一个Transaction注解。

你可能感兴趣的:(Java程序员进阶之路,java,数据库,开发语言)