共存

依赖

为了使用 Akka Actor 类型,你需要将以下依赖添加到你的项目中:

  1. <!-- Maven -->
  2. <dependency>
  3. <groupId>com.typesafe.akka</groupId>
  4. <artifactId>akka-actor-typed_2.12</artifactId>
  5. <version>2.5.23</version>
  6. </dependency>
  7. <!-- Gradle -->
  8. dependencies {
  9. compile group: 'com.typesafe.akka', name: 'akka-actor-typed_2.12', version: '2.5.23'
  10. }
  11. <!-- sbt -->
  12. libraryDependencies += "com.typesafe.akka" %% "akka-actor-typed" % "2.5.23"

简介

我们相信 Akka 类型(Typed)将逐渐在现有系统中被采用,因此在同一个ActorSystem中,能够同时使用类型化和非类型化的 Actor 是很重要的。此外,我们不能在一个大爆炸版本(one big bang release)中集成所有现有模块,这也是为什么这两种编写 Actor 的方式必须能够共存的另一个原因。

有两种不同的ActorSystemakka.actor.ActorSystemakka.actor.typed.ActorSystem

目前,类型化的 Actor 系统是通过在表面(hood)下使用非类型化的 Actor 系统来实现的。这在将来可能会改变。

类型化和非类型化可以通过以下方式交互:

  • 非类型化的 Actor 系统可以创建类型化的 Actor
  • 类型化的 Actor 可以向非类型化的 Actor 发送消息,反之亦然
  • 从非类型化的父级生成并监督类型化的子级,反之亦然
  • 从非类型化的 Actor 监视类型化的 Actor,反之亦然
  • 非类型化的 Actor 系统可以转换为类型化的 Actor 系统

示例使用非类型化类的完全限定类名来区分具有相同名称的类型化类和非类型化类。

非类型化到类型化

同时存在的应用程序可能仍然有一个非类型化的ActorSystem。这可以转换为类型化的ActorSystem,以便新代码和迁移的部件不依赖于非类型化的系统:

  1. // In java use the static methods on Adapter to convert from typed to untyped
  2. import akka.actor.typed.javadsl.Adapter;
  3. akka.actor.ActorSystem untypedActorSystem = akka.actor.ActorSystem.create();
  4. ActorSystem<Void> typedActorSystem = Adapter.toTyped(untypedActorSystem);

然后,对于新类型化的 Actor,这里介绍如何从未类型化的 Actor 创建、监视并向其发送消息。

  1. public abstract static class Typed {
  2. interface Command {}
  3. public static class Ping implements Command {
  4. public final akka.actor.typed.ActorRef<Pong> replyTo;
  5. public Ping(ActorRef<Pong> replyTo) {
  6. this.replyTo = replyTo;
  7. }
  8. }
  9. public static class Pong {}
  10. public static Behavior<Command> behavior() {
  11. return Behaviors.receive(Typed.Command.class)
  12. .onMessage(
  13. Typed.Ping.class,
  14. (context, message) -> {
  15. message.replyTo.tell(new Pong());
  16. return same();
  17. })
  18. .build();
  19. }
  20. }

顶级非类型化 Actor 是以通常的方式创建的:

  1. akka.actor.ActorSystem as = akka.actor.ActorSystem.create();
  2. akka.actor.ActorRef untyped = as.actorOf(Untyped.props());

然后它可以创建一个类型化的 Actor,监视它,并向它发送消息:

  1. public static class Untyped extends AbstractActor {
  2. public static akka.actor.Props props() {
  3. return akka.actor.Props.create(Untyped.class);
  4. }
  5. private final akka.actor.typed.ActorRef<Typed.Command> second =
  6. Adapter.spawn(getContext(), Typed.behavior(), "second");
  7. @Override
  8. public void preStart() {
  9. Adapter.watch(getContext(), second);
  10. second.tell(new Typed.Ping(Adapter.toTyped(getSelf())));
  11. }
  12. @Override
  13. public Receive createReceive() {
  14. return receiveBuilder()
  15. .match(
  16. Typed.Pong.class,
  17. message -> {
  18. Adapter.stop(getContext(), second);
  19. })
  20. .match(
  21. akka.actor.Terminated.class,
  22. t -> {
  23. getContext().stop(getSelf());
  24. })
  25. .build();
  26. }
  27. }

我们导入Adapter类并调用静态方法进行转换。

  1. // In java use the static methods on Adapter to convert from typed to untyped
  2. import akka.actor.typed.javadsl.Adapter;

要在类型化和非类型化之间转换,akka.actor.typed.javadsl.Adapter中有适配器方法。注意上面示例中的内联注释。

类型化到非类型化

让我们把这个例子颠倒过来,首先创建类型化的 Actor,然后再创建非类型化的 Actor 作为子级。

下面将演示如何创建、监视和向此非类型化 Actor 来回(back and forth)发送消息:

  1. public static class Untyped extends AbstractActor {
  2. public static akka.actor.Props props() {
  3. return akka.actor.Props.create(Untyped.class);
  4. }
  5. @Override
  6. public Receive createReceive() {
  7. return receiveBuilder()
  8. .match(
  9. Typed.Ping.class,
  10. message -> {
  11. message.replyTo.tell(new Typed.Pong());
  12. })
  13. .build();
  14. }
  15. }

创建 Actor 系统和类型化的 Actor:

  1. ActorSystem as = ActorSystem.create();
  2. ActorRef<Typed.Command> typed = Adapter.spawn(as, Typed.behavior(), "Typed");

然后,类型化的 Actor 创建非类型化的 Actor,监视它并发送和接收响应消息:

  1. public abstract static class Typed {
  2. public static class Ping {
  3. public final akka.actor.typed.ActorRef<Pong> replyTo;
  4. public Ping(ActorRef<Pong> replyTo) {
  5. this.replyTo = replyTo;
  6. }
  7. }
  8. interface Command {}
  9. public static class Pong implements Command {}
  10. public static Behavior<Command> behavior() {
  11. return akka.actor.typed.javadsl.Behaviors.setup(
  12. context -> {
  13. akka.actor.ActorRef second = Adapter.actorOf(context, Untyped.props(), "second");
  14. Adapter.watch(context, second);
  15. second.tell(
  16. new Typed.Ping(context.getSelf().narrow()), Adapter.toUntyped(context.getSelf()));
  17. return akka.actor.typed.javadsl.Behaviors.receive(Typed.Command.class)
  18. .onMessage(
  19. Typed.Pong.class,
  20. (_ctx, message) -> {
  21. Adapter.stop(context, second);
  22. return same();
  23. })
  24. .onSignal(akka.actor.typed.Terminated.class, (_ctx, sig) -> stopped())
  25. .build();
  26. });
  27. }
  28. }

监督

非类型化 Actor 的默认监督(supervision)是重新启动,而类型化 Actor 的默认监督是停止。当组合非类型化和类型化 Actor 时,默认监督基于子级的默认行为,即,如果非类型化 Actor 创建类型化子级,则其默认监督将停止;如果类型化的 Actor 创建了非类型化的子级,则其默认监督将是重新启动。


英文原文链接Coexistence.