声明式事务@Transaction失效场景

1. 开启事务支持

只需要通过 @EnableTransactionManagement 注解就可以开启声明式事务的支持,提供的可选值有:

  • proxyTargetClass:表示 AOP 代理是基于接口的还是基于类的,默认是基于接口代理。
  • mode:如果是有接口的话可以使用 PROXY ,如果没有接口的话可以使用 ASPECTJ 利用 CgLib 对类做增强。
  • order:指定事务拦截的顺序,默认是最低的优先级,这样可以保证其他的代理类是在事务开启之后执行。

2. @Transactional

开启了对事务注解的支持之后,我们就可以在需要的或者方法上面添加 @Transactional 注解,@Transactional 注解的话也有一些设置的项。

2.1 transactionManager

默认的话就是 DataSourceTransactionManager

2.2 propagation

事务的传播机制,默认的话是 REQUIRED ,多个方法调用共享同一个事务。

2.3 isolation

事务的隔离级别,这个跟使用的数据库有关系,默认的话是使用数据库默认的隔离级别。

2.4 timeout

事务超时时间。

2.5 readOnly

是否只读。

2.6 rollbackFor

或者方法中抛出的异常与 rollbackFor 中指定的异常 instanceof 返回 true 的情况下会回滚事务。

2.7 rollbackForClassName

rollbackFor 类似,只不过指定的是异常的名称。

2.8 noRollbackFor

遇到指定的异常时候不会回滚事务

2.9 noRollbackForClassName

noRollbackFor 类似,只不过指定的是异常的名称。

3. 没有通过代理对象执行

Spring 当中,声明式事务其实是为类做了一个代理,既然是代理类,也就是说需要去调用代理类才能够执行到需要被增强的方法,如果是在方法内部做调用的话,也就是没有走到代理方法或者说没有做增强处理,例如:

@Component
public class FooServiceImpl implements FooService {
  @Override
  @Transactional(rollbackFor = RollbackException.class)
  public void insertThenRollback() throws RollbackException {
    this.jdbcTemplate.execute("INSERT INTO FOO (BAR) values('BBB')");
    throw new RollbackException();
  }

  @Override
  public void invokeInsertThenRollback() throws RollbackException {
    this.insertThenRollback();
  }
}
@Override
public void run(String... args) throws Exception {
    try {
        this.fooService.invokeInsertThenRollback();
    } catch (RollbackException e) {
        // ignore.
    }
    log.info("bbb: {}.", this.countByBAR("BBB"));
}

日志输出如下:

bbb: 1.

要解决这个问题我们可以有以下几种方式:

3.1 通过当前代理对象调用自身实例

@Override
public void invokeInsertThenRollback() throws RollbackException {
  ((FooService) AopContext.currentProxy()).insertThenRollback();
}

3.2 注入自身实例

@Autowired
private FooService fooService;

@Override
public void invokeInsertThenRollback() throws RollbackException {
  this.fooService.insertThenRollback();
}

3.3 再加一层事务

@Override
@Transactional(rollbackFor = RollbackException.class)
public void invokeInsertThenRollback() throws RollbackException {
    this.insertThenRollback();
}

4. 不能对目标方法增强

@Transactional 只能支持 public 修饰方法,否则事务会失效,官方文档地址:Spring官方文档:声明式事务方法可见性

@Autowired
private FooServiceImpl fooService;

public void invokeInsertThenRollback() throws RollbackException {
    this.fooService.insertThenRollback();
}
@Transactional(rollbackFor = RollbackException.class)
protected void insertThenRollback() throws RollbackException {
    this.jdbcTemplate.execute("INSERT INTO FOO (BAR) values('BBB')");
    throw new RollbackException();
}

5. rollbackException与抛出的异常返回false

@Transactional 注解默认回滚异常是 RuntimeException 级别的异常,也就是非受检异常,如果是遇到了受检异常那么事务不会回滚:

public class RollbackException extends Exception {
}
@Transactional
public void insertThenRollback() throws RollbackException {
    this.jdbcTemplate.execute("INSERT INTO FOO (BAR) values('BBB')");
    throw new RollbackException();
}
public void invokeInsertThenRollback() throws RollbackException {
    this.fooService.insertThenRollback();
}

6. progagtion的选择问题

比如以下方法返回统计的 BBB1,有一条被回滚,有一条被成功执行。

@Override
@Transactional(rollbackFor = RollbackException.class,propagation = Propagation.REQUIRES_NEW)
public void insertThenRollback() {
    this.jdbcTemplate.execute("INSERT INTO FOO (BAR) values('BBB')");
}

@Override
@Transactional(rollbackFor = RollbackException.class)
public void invokeInsertThenRollback() throws RollbackException {
    this.jdbcTemplate.execute("INSERT into FOO (BAR) values('BBB')");
    this.fooService.insertThenRollback();
    throw new RollbackException();
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,258评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,335评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,225评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,126评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,140评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,098评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,018评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,857评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,298评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,518评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,678评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,400评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,993评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,638评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,801评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,661评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,558评论 2 352

推荐阅读更多精彩内容