Spring Data JPA @DomainEvents 发布领域事件以及遇到的坑。

发布领域事件

通过repositories管理的实体是聚合根。在领域驱动设计(DDD)的应用中,这些聚合根通常会发布领域事件。Spring Data提供了一个名为@DomainEvents的注解,可以在聚合根的一个方法上使用该注解,就可以非常方便的发布领域事件了。如下所示:

@Entity
class Person {
    private Long id;
    private String name;
    private int age;

    @Transient
    private final transient List<PersonEvent> domainEvents = new ArrayList();

    /**
	* 	使用 @DomainEvents 的方法可以返回一个单一的事件实例或一个事件的集合。
	*	它必须不接受任何参数。
	*/
    @DomainEvents
    Collection<PersonEvent> domainEvents() {
        return domainEvents;
    }

    /**
	* 在所有的事件都被发布后,可以使用@AfterDomainEventPublication注释的方法
    * 清理要发布的事件列表(除其他用途外)。
	*/
    @AfterDomainEventPublication 
    void callbackMethod() {
       domainEvents.clear();
    }

    public void saveDomainEvent(){
        domainEvents.add(new PersonSaveEvent(this));
    }

    public void deleteDomainEvent(){
        domainEvents.add(new PersonDeleteEvent(this));
    }
}

每次调用Spring Data Repository 的save(...)saveAll(...)delete(...)deleteAll(...)方法时都会调用这些方法,也就是增/删/改都会触发领域事件。

监听领域事件

使用注解 @TransactionalEventListener 来监听领域事件,如下所示为监听PersonSaveEvent事件

@Async
@TransactionalEventListener(phase = TransactionPhase.AFTER_COMMIT)
public void savePersonEvent(PersonSaveEvent saveEvent){
    this.doSomething(saveEvent.getPerson());
}

遇到的问题

  1. DomainEvents 方法不能使用 Arrays.asList 返回集合,会报错。
    Spring JPA @DomainEvents,从聚合根发布领域事件以及遇到的坑
    Spring JPA @DomainEvents,从聚合根发布领域事件以及遇到的坑。

出错原因:
Arrays.asList返回的集合是 Arrays 的内部类 ArrayList,查看源码发现ArrayList本身是没有实现remove方法的,所以会调用其父类 AbstractList remove 方法。
AbstractList 其实也没有实现 remove 方法,而是直接抛出个异常,也就是上面坑里抛出的错误。

  1. 直接在@DomainEvents标注的方法内创建指定类型事件。导致删除时也会调用saveEvent逻辑。
    Spring JPA @DomainEvents,从聚合根发布领域事件以及遇到的坑。
    Spring官方文档明确标注有-> 每次调用Spring Data Repository 的save(...)saveAll(...)delete(...)deleteAll(...)方法时都会触发领域事件。所以在domainEvents方法中根本就不能明确当前触发的到底是什么事件(创建 or 删除?)。

如需源码,请私信或留言。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

i余数

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值