美文网首页程序员
AxonFramework事件处理

AxonFramework事件处理

作者: 勇赴 | 来源:发表于2017-06-11 06:44 被阅读147次

saga中的事件处理非常接近一个普通的事件监听器。上述的对于方法和参数解析的规则在这里是有效的。不过,有一个主要区别。虽然存在事件监听器只有单个实例处理所有传入事件,但也存在一个saga有多个实例,每个实例都对不同的事件感兴趣。例如,关于Order的id为1的管理业务Saga对Order“2”的事件不感兴趣,反之亦然。

Axon不会将所有事件都发布给所有saga实例(这将是对资源的完全浪费),而是只发布与saga相关联的属性的事件。这个通过使用AssociationValues完成。一个AssociationValue由key和value组成。key代表标识符使用的类型,例如“orderId”或“order”。value表示前面例子中相应“1”或“2”值。

带@SagaEventHandler注解的方法被评估的顺序与带@EventHandler的相同。如果处理器方法的参数与传入的事件匹配,那么方法就匹配,如果saga有一个定义在处理器方法上的association属性。

@SagaEventHandler注解有两个attribute,其中associationProperty是最重要的。这是传入事件上property的名称,应该用来寻找相关的saga。association值的key是property的名称。这个值是由property的getter方法返回的值。

例如,考虑一个带”String getOrderId()”方法传入事件,返回“123”。如果一个带@SagaEventHandler(associationProperty = orderId)注解的方法接受这个事件,这个事件被路由到所有已经与带一个键为orderId和值为“123”的AssociationValues关联的saga。这可能是一个,多个,甚至没有。

有时,想要关联的属性的名称不是想要使用的关联的名称。例如,你有一个销售订单相匹配购买订单的saga。你可以有一个包含“buyOrderId”和“sellOrderId”的事务对象。如果你想要的saga将“orderId”作为关联的值,你可以定义一个不同的keyName 在@SagaEventHandler注解中。它将变成@SagaEventHandler(associationProperty="sellOrderId", keyName="orderId")。

管理关联
当一个saga事务管理跨多个域的概念,如订单、发货,、发票,等等,saga需要与这些概念的实例关联。一个关联需要两个参数:key,识别关联的类型(订单、发货等)和一个value,该值表示该概念的标识符。
在以下几个方面完成用概念关联到saga。第一,当新创建一个saga时将调用一个用@StartSaga注解的事件处理器时,它将自动与@SagaEventHandler方法中标识的关联。所有其他的关联用SagaLifecycle.associateWith(String key, String/Number value)方法创建。用SagaLifecycle.removeAssociationWith(String key, String/Number value)方法去移除一个特定的关联。

想象一下为一个围绕着订单的事务而已经被创建的一个saga。saga自动关联订单,方法被@StartSaga注解。saga是负责创建该订单的发票,并告诉航运创建一个载货量。一旦货物到达和发票支付,交易完成后,saga被关闭。
这是一个saga的代码:

public class OrderManagementSaga {
private boolean paid = false;
private boolean delivered = false;
@Inject
private transient CommandGateway commandGateway;

@StartSaga
@SagaEventHandler(associationProperty = "orderId")
public void handle(OrderCreatedEvent event) {
    // client generated identifiers
    ShippingId shipmentId = createShipmentId();
    InvoiceId invoiceId = createInvoiceId();
    // associate the Saga with these values, before sending the commands
    associateWith("shipmentId", shipmentId);
    associateWith("invoiceId", invoiceId);
    // send the commands
    commandGateway.send(new PrepareShippingCommand(...));
    commandGateway.send(new CreateInvoiceCommand(...));
}

@SagaEventHandler(associationProperty = "shipmentId")
public void handle(ShippingArrivedEvent event) {
    delivered = true;
    if (paid) { end(); }
}

@SagaEventHandler(associationProperty = "invoiceId")
public void handle(InvoicePaidEvent event) {
    paid = true;
    if (delivered) { end(); }
}

// ...
}

通过允许客户端生成标识符,可以很容易地与一个概念相关联,而不需要请求响应类型命令。在发布命令之前,我们将事件与这些概念关联起来。通过这种方式,我们也保证捕捉到作为该命令的一部分生成的事件。一旦发票付清,货物到达,saga也将结束。

相关文章

  • AxonFramework事件处理

    saga中的事件处理非常接近一个普通的事件监听器。上述的对于方法和参数解析的规则在这里是有效的。不过,有一个主要区...

  • AxonFramework,事件发布和处理

    由应用程序生成的事件需要被分发到更新查询数据库的组件,搜索引擎或其他需要它们的资源:事件处理程序(Event Ha...

  • AxonFramework,分发事件

    在某些情况下,有必要发布事件到外部系统,比如消息broker。 Spring AMQP Axon提供了开箱即用的支...

  • AxonFramework聚合事件溯源

    除了存储一个聚合的当前状态,还可以根据过去发布的事件恢复一个聚合的状态。为此,所有状态的更改必须由一个事件来表示。...

  • AxonFramework消息、命令和事件

    消息传送的概念消息是Axon的核心概念之一。组件之间的所有通信都使用消息对象完成。这为这些组件提供了位置透明性,在...

  • AxonFramework,事件向上转换(Event Upcas

    由于软件应用程序的不断变化的性质,很可能事件定义也随着时间而变化。由于事件存储被认为是只读和只追加(没有修改和删除...

  • AxonFramework在聚合中处理命令

    建议在包含处理状态命令的聚合中直接定义命令处理器,因为命令处理器有可能需要该集合的状态来执行其任务。 要在一个聚合...

  • JS 事件

    目录 事件流 事件处理程序HTML事件处理程序DOM0级事件处理程序DOM2级事件处理程序IE事件处理程序跨浏览器...

  • 如何构建一个交易系统(九)

    AxonFramework是我们交易系统选择的架构基础, 使用CQRS/EventSource 不拘泥于框架使用,...

  • react事件处理

    一,事件处理 写法:on+事件名称= {事件处理函数} 类组件触发写法on+事件名称 = 事件处理函数 ---...

网友评论

    本文标题:AxonFramework事件处理

    本文链接:https://www.haomeiwen.com/subject/esxhqxtx.html