vertx-verticle介绍

什么是Verticle ?

Vert.x 通过开箱即用的方式提供了一个简单便捷的、可扩展的、类似 Actor Model 的部署和并发模型机制。您可以用此模型机制来保管您自己的代码组件。

这个模型是可选的,如果您不想这样做,Vert.x 不会强迫您用这种方式创建您的应用程序。

这个模型不能说是严格的 Actor 模式的实现,但它确实有相似之处,特别是在并发、扩展性和部署等方面。

要使用该模型,您需要将您的代码组织成一系列的 Verticle。

Verticle 是由 Vert.x 部署和运行的代码块。默认情况一个 Vert.x 实例维护了N(默认情况下N = CPU核数 x 2)个 Event Loop 线程。Verticle 实例可使用任意 Vert.x 支持的编程语言编写,而且一个简单的应用程序也可以包含多种语言编写的 Verticle。

您可以将 Verticle 想成 Actor Model 中的 Actor。

一个应用程序通常是由在同一个 Vert.x 实例中同时运行的许多 Verticle 实例组合而成。不同的 Verticle 实例通过向 Event Bus 上发送消息来相互通信。

Verticle 核心方法

  • getVertx()
    该方法用来获取该verticle实例发布到的Vertx对象。
  • void init(Vertx vertx, Context context);
    初始化一个Verticle实例。
  • void start(Future startFuture) throws Exception;
    启动化一个Verticle实例。Vert.x在发布Verticle的时候会调用该方法,在应用代码中不应该调用该方法。
  • void stop(Future stopFuture) throws Exception;
    停止一个Verticle, Vert.x在撤销一个Verticle的时候会调用该方法。

编写 Verticle

Verticle的实现类必须实现Verticle接口。

如果您喜欢的话,可以直接实现该接口,但是通常直接从抽象类AbstractVerticle继承更简单。

这儿有一个例子:

1
2
3
4
5
6
7
8
public class MyVerticle extends AbstractVerticle {
// Called when verticle is deployed
public void start() {
}
// Optional - called when verticle is undeployed
public void stop() {
}
}

通常你需要像上边例子一样重写 start 方法。

当 Vert.x 部署 Verticle 时,它的start方法将被调用,这个方法执行完成后 Verticle 就变成已启动状态。

你同样可以重写stop方法,当Vert.x 撤销一个 Verticle 时它会被调用,这个方法执行完成后 Verticle 就变成已停止状态了。

Verticle 异步启动和停止

有些时候你的Verticle启动可能会依赖其它Verticle的启动结果,如:你想在start方法中部署其他的Verticle

你不能在你的start方法中阻塞等待其他的Verticle部署完成,这样做会破坏 黄金法则。

所以你要怎么做?

你可以实现异步版本start方法来做这个事。这个版本的方法会以一个Future作参数被调用。start方法执行完时,Verticle实例并没有部署好(状态不是deployed)。稍后,依赖的事情完成以后(如:启动其他Verticle),你就可以调用Futurecomplete(或fail)方法来标记启动完成或失败了。

这儿有一个例子:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
public class MyVerticle extends AbstractVerticle {

@Override
public void start(Future<Void> startFuture) {
// 现在部署其他的一些verticle
vertx.deployVerticle("com.foo.OtherVerticle", res -> {
if (res.succeeded()) {
startFuture.complete();
} else {
startFuture.fail(res.cause());
}
});
}
}

同样的,这儿也有一个异步版本的 stop 方法,如果您想做一些耗时的 Verticle 清理工作,你可以使用它

1
2
3
4
5
6
7
8
9
10
11
12
public class MyVerticle extends AbstractVerticle {

public void stop(Future<Void> stopFuture) {
obj.doSomethingThatTakesTime(res -> {
if (res.succeeded()) {
stopFuture.complete();
} else {
stopFuture.fail();
}
});
}
}

请注意:不需要在一个 Verticle 的stop方法中手工去撤销启动时部署的子 Verticle,当父 Verticle 在撤销时 Vert.x 会自动撤销任何子 Verticle。

Verticle 种类

这儿有三种不同类型的 Verticle:

  • Stardand Verticle:这是最常用的一类 Verticle —— 它们永远运行在 Event Loop 线程上。稍后的章节我们会讨论更多。
  • Worker Verticle:这类 Verticle 会运行在 Worker Pool 中的线程上。一个实例绝对不会被多个线程同时执行。
  • Multi-Threaded Worker Verticle:这类 Verticle 也会运行在 Worker Pool 中的线程上。一个实例可以由多个线程同时执行(译者注:因此需要开发者自己确保线程安全)。

Standard Verticle

Standard Verticle 被创建时,它会被分派给一个 Event Loop 线程,并在这个 Event Loop 中执行它的start方法。当您在一个 Event Loop 上调用了 Core API 中的方法并传入了处理器时,Vert.x 将保证用与调用该方法时相同的 Event Loop 来执行这些处理器。

这意味着我们可以保证您的 Verticle 实例中 所有的代码都是在相同Event Loop中执行(只要您不创建自己的线程并调用它!)

同样意味着您可以将您的应用中的所有代码用单线程方式编写,让 Vert.x 去考虑线程和扩展问题。您不用再考虑 synchronized 和 volatile 的问题,也可以避免传统的多线程应用经常会遇到的竞态条件和死锁的问题。

Worker Verticle

Worker VerticleStandard Verticle 很像,但它并不是由一个 Event Loop 来执行,而是由Vert.x中的 Worker Pool 中的线程执行。

Worker Verticle 被设计来调用阻塞式代码,它不会阻塞任何 Event Loop

如果您不想使用 Worker Verticle 来运行阻塞式代码,您还可以在一个Event Loop中直接使用 内联阻塞式代码

若您想要将 Verticle 部署成一个 Worker Verticle,您可以通过 setWorker 方法来设置:

1
2
DeploymentOptions options = new DeploymentOptions().setWorker(true);
vertx.deployVerticle("com.mycompany.MyOrderProcessorVerticle", options);

Worker Verticle 实例绝对不会在 Vert.x 中被多个线程同时执行,但它可以在不同时间由不同线程执行。

Multi-threaded Worker Verticle

一个 Multi-threaded Worker Verticle 近似于普通的 Worker Verticle,但是它可以由不同的线程同时执行。

警告:Multi-threaded Worker Verticle 是一个高级功能,大部分应用程序不会需要它。由于这些 Verticle 是并发的,您必须小心地使用标准的Java多线程技术来保持 Verticle 的状态一致性。

编程方式部署Verticle

您可以指定一个Verticle名称或传入您已经创建好的Verticle实例,使用任意一个deployVerticle方法来部署Verticle。

请注意:通过 Verticle 实例 来部署 Verticle 仅限Java语言。

1
2
Verticle myVerticle = new MyVerticle();
vertx.deployVerticle(myVerticle);

您同样可以指定 Verticle 的名称来部署它。

这个 Verticle 的名称会用于查找实例化 Verticle 的特定VerticleFactory

不同的VerticleFactory 可用于实例化不同语言的 Verticle,也可用于其他目的,例如:加载服务、运行时从Maven中获取Verticle实例等。

这允许您部署用任何使用Vert.x支持的语言编写的Verticle实例。

这儿有一个部署不同类型 Verticle 的例子:

1
2
3
4
5
6
7
vertx.deployVerticle("com.mycompany.MyOrderProcessorVerticle");

// 部署JavaScript的Verticle
vertx.deployVerticle("verticles/myverticle.js");

// 部署Ruby的Verticle
vertx.deployVerticle("verticles/my_verticle.rb");

Verticle名称到Factory的映射规则

当使用名称部署Verticle时,会通过名称来选择一个用于实例化 Verticle 的 Verticle Factory。

Verticle 名称可以有一个前缀 —— 使用字符串紧跟着一个冒号,它用于查找存在的Factory,参考例子

1
2
3
js:foo.js // 使用JavaScript的Factory
groovy:com.mycompany.SomeGroovyCompiledVerticle // 用Groovy的Factory
service:com.mycompany:myorderservice // 用Service的Factory

如果不指定前缀,Vert.x将根据提供名字后缀来查找对应Factory,如:

1
2
foo.js // 将使用JavaScript的Factory
SomeScript.groovy // 将使用Groovy的Factory

若前缀后缀都没指定,Vert.x将假定这个名字是一个Java 全限定类名(FQCN)然后尝试实例化它。

如何定位Verticle Factory?

大部分VerticleFactory会从classpath中加载,并在 Vert.x 启动时注册。

您同样可以使用编程的方式去注册或注销VerticleFactory:通过registerVerticleFactory 方法和unregisterVerticleFactory 方法。

等待部署完成

Verticle的部署是异步方式,可能在 deploy 方法调用返回后一段时间才会完成部署。

如果您想要在部署完成时被通知则可以指定一个完成处理器:

1
2
3
4
5
6
7
vertx.deployVerticle("com.mycompany.MyOrderProcessorVerticle", res -> {
if (res.succeeded()) {
System.out.println("Deployment id is: " + res.result());
} else {
System.out.println("Deployment failed!");
}
});

如果部署成功,这个完成处理器的结果中将会包含部署ID的字符串。这个部署 ID可以在之后您想要撤销它时使用。

撤销Verticle

我们可以通过 undeploy 方法来撤销部署好的 Verticle。

撤销操作也是异步的,因此若您想要在撤销完成过后收到通知则可以指定另一个完成处理器:

1
2
3
4
5
6
7
vertx.undeploy(deploymentID, res -> {
if (res.succeeded()) {
System.out.println("Undeployed ok");
} else {
System.out.println("Undeploy failed!");
}
});

设置 Verticle 实例数

当使用名称部署一个 Verticle 时,您可以指定您想要部署的 Verticle 实例的数量。

1
2
DeploymentOptions options = new DeploymentOptions().setInstances(16);
vertx.deployVerticle("com.mycompany.MyOrderProcessorVerticle", options);

向 Verticle 传入配置

可在部署时传给 Verticle 一个 JSON 格式的配置

1
2
3
JsonObject config = new JsonObject().put("name", "tim").put("directory", "/blah");
DeploymentOptions options = new DeploymentOptions().setConfig(config);
vertx.deployVerticle("com.mycompany.MyOrderProcessorVerticle", options);

传入之后,这个配置可以通过Context对象或使用config方法访问。

这个配置会以 JSON 对象(JsonObject)的形式返回,因此您可以用下边代码读取数据:

1
System.out.println("Configuration: " + config().getString("name"));

在 Verticle 中访问环境变量

环境变量和系统属性可以直接通过 Java API 访问:

1
2
System.getProperty("prop");
System.getenv("HOME");

Verticle 隔离组

默认情况,当Vert.x部署Verticle时它会调用当前类加载器来加载类,而不会创建一个新的。大多数情况下,这是最简单、最清晰和最干净。

但是在某些情况下,您可能需要部署一个Verticle,它包含的类要与应用程序中其他类隔离开来。比如您想要在一个Vert.x实例中部署两个同名不同版本的Verticle,或者不同的Verticle使用了同一个jar包的不同版本。

当使用隔离组时,您需要用setIsolatedClassed方法来提供一个您想隔离的类名列表。列表项可以是一个Java 限定类全名,如 com.mycompany.myproject.engine.MyClass;也可以是包含通配符的可匹配某个包或子包的任何类,例如 com.mycompany.myproject.* 将会匹配所有 com.mycompany.myproject 包或任意子包中的任意类名。

请注意仅仅只有匹配的类会被隔离,其他任意类会被当前类加载器加载。

若您想要加载的类和资源不存在于主类路径(main classpath),您可使用 setExtraClasspath 方法将额外的类路径添加到这里。

警告:谨慎使用此功能,类加载器可能会导致您的应用难于调试,变得一团乱麻(can of worms)。

以下是使用隔离组隔离 Verticle 的部署例子:

1
2
3
4
DeploymentOptions options = new DeploymentOptions().setIsolationGroup("mygroup");
options.setIsolatedClasses(Arrays.asList("com.mycompany.myverticle.*",
"com.mycompany.somepkg.SomeClass", "org.somelibrary.*"));
vertx.deployVerticle("com.mycompany.myverticle.VerticleClass", options);

高可用性

Verticle可以启用高可用方式(HA)部署。在这种方式下,当其中一个部署在 Vert.x 实例中的 Verticle 突然挂掉,这个 Verticle 可以在集群环境中的另一个 Vert.x 实例中重新部署。

若要启用高可用方式运行一个 Verticle,仅需要追加 -ha 参数:

1
vertx run my-verticle.js -ha

当启用高可用方式时,不需要追加 -cluster 参数。

关于高可用的功能和配置的更多细节可参考 高可用和故障转移 章节。

从命令行运行Verticle

您可以在 Maven 或 Gradle 项目中以正常方式添加 Vert.x Core 为依赖,在项目中直接使用 Vert.x。

但是,您也可以从命令行直接运行 Vert.x 的 Verticle。

为此,您需要下载并安装 Vert.x 的发行版,并且将安装的 bin 目录添加到您的 PATH 环境变量中,还要确保您的 PATH 中设置了Java 8的JDK环境。

请注意:JDK需要支持Java代码的运行时编译(on the fly compilation)。

现在您可以使用 vertx run 命令运行Verticle了,这儿是一些例子:

1
2
3
4
5
6
7
8
# 运行JavaScript的Verticle
vertx run my_verticle.js

# 运行Ruby的Verticle
vertx run a_n_other_verticle.rb

# 使用集群模式运行Groovy的Verticle
vertx run FooVerticle.groovy -cluster

您甚至可以不必编译 Java 源代码,直接运行它:

1
vertx run SomeJavaSourceFile.java

Vert.x 将在运行它之前对 Java 源代码文件执行运行时编译,这对于快速原型制作和演示很有用。不需要设置 Maven 或 Gradle 就能跑起来!

欲了解有关在命令行执行 vertx 可用的各种选项完整信息,可以直接在命令行键入 vertx 查看帮助。

退出 Vert.x 环境

Vert.x 实例维护的线程不是守护线程,因此它们会阻止JVM退出。

如果您通过嵌入式的方式使用 Vert.x 并且完成了操作,您可以调用 close 方法关闭它。这将关闭所有内部线程池并关闭其他资源,允许JVM退出。

文章目录
  1. 1. 什么是Verticle ?
  2. 2. Verticle 核心方法
  3. 3. 编写 Verticle
  4. 4. Verticle 异步启动和停止
  5. 5. Verticle 种类
    1. 5.1. Standard Verticle
    2. 5.2. Worker Verticle
    3. 5.3. Multi-threaded Worker Verticle
  6. 6. 编程方式部署Verticle
  7. 7. Verticle名称到Factory的映射规则
  8. 8. 如何定位Verticle Factory?
  9. 9. 等待部署完成
  10. 10. 撤销Verticle
  11. 11. 设置 Verticle 实例数
  12. 12. 向 Verticle 传入配置
  13. 13. 在 Verticle 中访问环境变量
  14. 14. Verticle 隔离组
  15. 15. 高可用性
  16. 16. 从命令行运行Verticle
  17. 17. 退出 Vert.x 环境
|