vertx 异步编程指南 step8-使用RxJava进行反应式编程

2018-04-23 13:15:32 zyydecsdn 阅读数 1212  收藏 更多

分类专栏: vertx
 

到目前为止,我们已经使用基于回调的API探索了Vert.x堆栈的各个领域。它只是起作用,并且这种编程模型在很多语言的开发人员中都是众所周知的。然而,它可能会变得有点乏味,特别是当你结合几个事件源或处理复杂的数据流时。

这正是RxJava发挥的作用,Vert.x可以无缝集成它。

注意
在本指南中,使用了RxJava 2.x,但Vert.x也适用于RxJava 1.x. RxJava 2.x在Reactive-Streams规范的基础上完全重写。在2.0 wiki页面的不同之处了解更多信息。

启用RxJava API

除了基于回调的API之外,Vert.x模块还提供了“Rxified” API。要启用它,首先将vertx-rx-java2模块添加到Maven POM文件中:

<dependency>

<groupId>io.vertx</groupId>

<artifactId>vertx-rx-java2</artifactId>

</dependency>

Verticle必须进行修改,以便扩展io.vertx.reactivex.core.AbstractVerticle而不是io.vertx.core.AbstractVerticle。这有什么不同?前一类扩展后者并暴露一个io.vertx.reactivex.core.Vertx领域。

io.vertx.reactivex.core.Vertx定义rxSomething(…​)与基于回调的对应方法等效的额外方法。

让我们来看看MainVerticle如何更好地了解它在实践中的工作原理:

Single<String> dbVerticleDeployment = vertx.rxDeployVerticle(

"io.vertx.guides.wiki.database.WikiDatabaseVerticle");

rxDeploy方法不会将Handler<AsyncResult<String>>最终参数作为参数。相反,它返回一个Single<String>

此外,调用该方法时操作不会启动。它在你订阅时开始Single。操作完成后,它会发出部署id或使用a指示问题的原因Throwable

按顺序部署Verticle

要最终确定MainVerticle重构,我们必须确保部署操作按顺序触发并发生:

dbVerticleDeployment
.flatMap(id -> { (1)
Single<String> httpVerticleDeployment = vertx.rxDeployVerticle(
"io.vertx.guides.wiki.http.HttpServerVerticle",
new DeploymentOptions().setInstances(2));
return httpVerticleDeployment;
}) .subscribe(id -> startFuture.complete(), startFuture::fail); (2)
  1. flatMap运营商应用功能的结果dbVerticleDeployment。它在这里安排的部署HttpServerVerticle

  2. 订阅时开始操作。在成功或错误时,MainVerticle开始未来可能完成或失败。

部分“Rxifying” HttpServerVerticle

如果按顺序跟随指南,随时编辑代码,那么HttpServerVerticle类仍然使用基于回调的API。在您可以使用RxJava API 自然执行异步操作之前,即 同时需要重构HttpServerVerticle

导入Vert.x类的RxJava版本

import io.vertx.reactivex.core.AbstractVerticle;

import io.vertx.reactivex.core.http.HttpServer;

import io.vertx.reactivex.ext.auth.AuthProvider;

import io.vertx.reactivex.ext.auth.User;

import io.vertx.reactivex.ext.auth.jwt.JWTAuth;

import io.vertx.reactivex.ext.auth.shiro.ShiroAuth;

import io.vertx.reactivex.ext.web.Router;

import io.vertx.reactivex.ext.web.RoutingContext;

import io.vertx.reactivex.ext.web.client.WebClient;

import io.vertx.reactivex.ext.web.client.HttpResponse; (1)

import io.vertx.reactivex.ext.web.codec.BodyCodec;

import io.vertx.reactivex.ext.web.handler.*;

import io.vertx.reactivex.ext.web.sstore.LocalSessionStore;

import io.vertx.reactivex.ext.web.templ.FreeMarkerTemplateEngine;

import org.slf4j.Logger;

import org.slf4j.LoggerFactory;
  1. 我们的backupHandler()方法仍然使用HttpResponse类,所以它必须导入。事实证明,HttpResponseVert.x提供的RxJava版本可以作为这种特定情况下的替代品。所述“Rxified”在代码step-8作为响应类型由lambda表达式推断引导库的文件夹不导入此类。

 

“Rxified” vertx实例上使用委托

要调用一个方法,io.vertx.core.Vertx当你有一个方法时io.vertx.reactivex.core.Vertx,调用该getDelegate()方法。start()在创建以下实例时需要调整Verticle的方法 WikiDatabaseService

  1.  
    @Override
  2.  
    public void start(Future<Void> startFuture) throws Exception {
  3.  
     
  4.  
    String wikiDbQueue = config().getString(CONFIG_WIKIDB_QUEUE, "wikidb.queue");
  5.  
    dbService = io.vertx.guides.wiki.database.WikiDatabaseService.createProxy(vertx.getDelegate(), wikiDbQueue);

同时执行授权查询

在前面的例子中,我们看到了如何使用RxJava操作符和Rxified Vert.x API按顺序执行异步操作。但有时候这种担保不是必需的,或者您只是希望它们为了性能原因而同时运行。

JWT令牌生成过程HttpServerVerticle就是这种情况的一个很好的例子。要创建令牌,我们需要完成所有授权查询,但查询彼此独立:

  1.  
    auth.rxAuthenticate(creds).flatMap(user -> {
  2.  
    Single<Boolean> create = user.rxIsAuthorised("create"); (1)
  3.  
    Single<Boolean> delete = user.rxIsAuthorised("delete");
  4.  
    Single<Boolean> update = user.rxIsAuthorised("update");
  5.  
     
  6.  
    return Single.zip(create, delete, update, (canCreate, canDelete, canUpdate) -> { (2)
  7.  
    return jwtAuth.generateToken(
  8.  
    new JsonObject()
  9.  
    .put("username", context.request().getHeader("login"))
  10.  
    .put("canCreate", canCreate)
  11.  
    .put("canDelete", canDelete)
  12.  
    .put("canUpdate", canUpdate),
  13.  
    new JWTOptions()
  14.  
    .setSubject("Wiki API")
  15.  
    .setIssuer("Vert.x"));
  16.  
    });
  17.  
    }).subscribe(token -> {
  18.  
    context.response().putHeader("Content-Type", "text/plain").end(token);
  19.  
    }, t -> context.fail(401));
  1. Single创建三个对象,表示不同的授权查询。

  2. 当三个操作成功完成时,zip运算符回调将与结果一起调用。

查询数据库

直接查询

通常,需要单个数据库查询来准备对用户的响应。对于这样简单的情况,JDBCClient提供rxQueryXXXrxUpdateXXX方法:

  1.  
    String query = sqlQueries.get(SqlQuery.GET_PAGE_BY_ID);
  2.  
    JsonArray params = new JsonArray().add(id);
  3.  
    Single<ResultSet> resultSet = dbClient.rxQueryWithParams(query, params);

使用数据库连接

当直接查询不适合时(例如,当多个查询必须参与相同的事务时),您可以从池中获取数据库连接。所有你需要做的是呼吁rxGetConnectionJDBCClient

Single<SQLConnection> connection = dbClient.rxGetConnection();

该方法返回一个Single<Connection>您可以轻松转换flatMapXXX以执行SQL查询的方法:

  1.  
    connection
  2.  
    .flatMapCompletable(conn -> conn.rxExecute(sqlQueries.get(SqlQuery.CREATE_PAGES_TABLE)))

但是如果SQLConnection参考不再可达,我们怎么能释放连接呢?一个简单而方便的方法是closedoFinally回调中调用:

  1.  
    private Single<SQLConnection> getConnection() {
  2.  
    return dbClient.rxGetConnection().flatMap(conn -> {
  3.  
    Single<SQLConnection> connectionSingle = Single.just(conn); (1)
  4.  
    return connectionSingle.doFinally(conn::close); (2)
  5.  
    });
  6.  
    }
  1. 在获得连接后,我们将其包装成一个 Single

  2. Single被修改,以调用close一个doFinally回调

现在我们将getConnection随时使用我们需要的数据库连接。

缩小回调和RxJava之间的差距

有时,您可能必须将RxJava代码与基于回调的API混合使用。例如,服务代理接口只能用回调来定义,但实现使用Vert.x Rxified API。

在这种情况下,io.vertx.reactivex.SingleHelper.toObserver类可以适应Handler<AsyncResult<T>>RxJava SingleObserver<T>

  1.  
    @Override
  2.  
    public WikiDatabaseService fetchAllPagesData(Handler<AsyncResult<List<JsonObject>>> resultHandler) { (1)
  3.  
    dbClient.rxQuery(sqlQueries.get(SqlQuery.ALL_PAGES_DATA))
  4.  
    .map(ResultSet::getRows)
  5.  
    .subscribe(SingleHelper.toObserver(resultHandler)); (2)
  6.  
    return this;
  7.  
    }
  1. fetchAllPagesData是一种异步服务代理操作,用Handler<AsyncResult<List<JsonObject>>>回调定义。

  2. toObserver方法适用resultHandler于a SingleObserver<List<JsonObject>>,以便在发布行列表时调用处理程序。

注意
io.vertx.reactivex.CompletableHelperio.vertx.reactivex.MaybeHelper提供适配器CompletableMaybe

数据流

RxJava不仅善于组合不同的事件源,而且对数据流也非常有帮助。与Vert.x或JDK未来不同,a Flowable不仅发布一个事件流,而且还发布一系列事件。它配备了大量的数据操作操作符。

我们可以使用其中的一些来重构fetchAllPages数据库垂直方法:

  1.  
    public WikiDatabaseService fetchAllPages(Handler<AsyncResult<JsonArray>> resultHandler) {
  2.  
    dbClient.rxQuery(sqlQueries.get(SqlQuery.ALL_PAGES))
  3.  
    .flatMapPublisher(res -> { (1)
  4.  
    List<JsonArray> results = res.getResults();
  5.  
    return Flowable.fromIterable(results); (2)
  6.  
    })
  7.  
    .map(json -> json.getString(0)) (3)
  8.  
    .sorted() (4)
  9.  
    .collect(JsonArray::new, JsonArray::add) (5)
  10.  
    .subscribe(SingleHelper.toObserver(resultHandler));
  11.  
    return this;
  12.  
    }
  1. flatMapPublisher我们将创建一个Flowable从发射的项目Single<Result>

  2. fromIterable将数据库结果Iterable转换为Flowable发出数据库行项目。

  3. 由于我们只需要页面名称,我们可以将map每一JsonObject行记录到第一列。

  4. 客户希望数据sorted按字母顺序排列。

  5. 事件巴士服务答复包含在一个单一的JsonArraycollect创建一个新的JsonArray::new项目,随后添加项目JsonArray::add

vertx 异步编程指南 step8-使用RxJava进行反应式编程的更多相关文章

  1. Apache Spark 2.2.0 中文文档 - Structured Streaming 编程指南 | ApacheCN

    Structured Streaming 编程指南 概述 快速示例 Programming Model (编程模型) 基本概念 处理 Event-time 和延迟数据 容错语义 API 使用 Data ...

  2. RxJava(一):响应式编程与Rx

    一,响应式编程 响应式编程是一种关注于数据流(data streams)和变化传递(propagation of change)的异步编程方式. 1.1 异步编程 传统的编程方式是顺序执行的,必须在完 ...

  3. 响应式编程库RxJava初探

    引子 在读 Hystrix 源码时,发现一些奇特的写法.稍作搜索,知道使用了最新流行的响应式编程库RxJava.那么响应式编程究竟是怎样的呢? 本文对响应式编程及 RxJava 库作一个初步的探索. ...

  4. Swift 响应式编程 浅析

    这里我讲一下响应式编程(Reactive Programming)是如何将异步编程推到一个全新高度的. 异步编程真的很难 大多数有关响应式编程的演讲和文章都是在展示Reactive框架如何好如何惊人, ...

  5. SpringBoot 2.x (14):WebFlux响应式编程

    响应式编程生活案例: 传统形式: 一群人去餐厅吃饭,顾客1找服务员点餐,服务员把订单交给后台厨师,然后服务员等待, 当后台厨师做好饭,交给服务员,经过服务员再交给顾客1,依此类推,该服务员再招待顾客2 ...

  6. Reactive UI -- 反应式编程UI框架入门学习(一)

    推荐一个反应式编程的MVVM跨平台框架. 反应式编程 反应式编程是一种相对于命令式的编程范式,由函数式的组合声明来构建异步数据流.要理解这个概念,可以简单的借助Excel中的单元格函数. 上图中,A1 ...

  7. 函数响应式编程(FRP)—基础概念篇

    原文出处:http://ios.jobbole.com/86815/. 一函数响应式编程 说到函数响应式编程,就不得不提到函数式编程,他们俩有什么关系呢?今天我们就详细的解析一下他们的关系. 现在下面 ...

  8. jQuery编程基础精华01(jQuery简介,顶级对象$,jQuery对象、Dom对象,链式编程,选择器)

    jQuery简介 什么是jQuery? jQuery就是一个JavaScript函数库,没什么特别的.(开源)联想SQLHelper类 jQuery能做什么?jQuery是做什么的? jQuery本身 ...

  9. iOS开发技巧系列---使用链式编程和Block来实现UIAlertView

    UIAlertView是iOS开发过程中最常用的控件之一,是提醒用户做出选择最主要的工具.在iOS8及后来的系统中,苹果更推荐使用UIAlertController来代替UIAlertView.所以本 ...

随机推荐

  1. linux设备驱动程序-设备树(2)-device_node转换成platform_device

    设备树处理之--device_node转换成platform_device 以下讨论基于linux4.14,arm平台 platform device 设备树的产生就是为了替代driver中过多的pl ...

  2. docker研究-5 docker网络介绍

    例子:启动(创建)一个容器,自定义容器名字为my_nginxtest02,镜像为nginx,将宿主机(本机)81端口映射到容器的80端口 [root@localhost ~]# docker run ...

  3. ubuntu 16.04中limit 修改

    第一,修改/etc/security/limits.conf: * soft nproc 65535* hard nproc 65535* soft nofile 65535* hard nofile ...

  4. 解决chrome连接自建https服务器报“您的连接不是私密连接”问题

    前一段时间,Chrome 突然显示出了“您的连接不是私密连接”,这下可难受了,大部分的网站打开都有问题. 找了各种方法,各种设置都是不行. 一.暴力.费力的方法直接卸载 Chrome ,删除一切数据以 ...

  5. MyBatisPlus快速入门

    MyBatisPlus快速入门 官方网站 https://mp.baomidou.com/guide 慕课网视频 https://www.imooc.com/learn/1130 入门 https:/ ...

  6. Nginx配置文件nginx.conf(八)

    原文链接:https://www.cnblogs.com/knowledgesea/p/5175711.html 在nginx.conf的注释符号是#. 默认的nginx.conf内容为: #user ...

  7. oracle左连接与右连接

    left join(左联接)       ---返回左表中的所有记录和右表中条件字段相等的记录. right join(右联接)     ---返回右表中的所有记录和左表中联结字段相等的记录 inne ...

  8. 微信小程序实现图片放大预览效果

    可以直接用微信程序自己的api很方便的实现 核心方法 wx.previewImage: 直接上代码, wxml: <!--pages/prewpicture/prew.wxml--> &l ...

  9. docker--发布docker镜像

    前戏 前面我们自己做了个docker镜像,我们可以上传到docker hub,别人就可以下载使用了 发布到docker hub 我们前面使用docker search 查找的镜像都是从docker h ...

  10. MySQL实战45讲学习笔记:第八讲

    一.今日内容概要 我在第 3 篇文章和你讲事务隔离级别的时候提到过,如果是可重复读隔离级别,事务 T 启动的时候会创建一个视图 read-view,之后事务 T 执行期间,即使有其他事务修改了数据,事 ...