您好,我是湘王,这是我的博客园,欢迎您来,欢迎您再来~

前面把Java函数式编程的由来和最主要的核心知识点讲完了。包括比较难懂的Lambda表达式是怎么演变而来的也全部都撸了一遍。Lambda表达式这种编程方式的确是让人不太习惯,尤其是之前那种纯面向对象编程的思维模式一旦建立起来之后,想要再接受这种比较奇怪的语法和编程模式,确实是非常困难。但Lambda表达式和流式编程的出现,又催生了另一门新的技术:反应式编程。

《三国演义》中说:“天下大势,分久必合,合久必分”。在科技领域也是一样。之前没有科学的时候,由占星术、炼金术、神学等催生出了一系列的自然科学,像什么物理、化学、数学等等。等到这些学科演变的差不多了之后,又开始了逐步的融合。

反应式编程就和这有点类似,它是融合了Lambda表达式、流式编程和观察者模式的一种新的编程范式。

就像很多反应式编程的技术书、官网和资料里面说的,“一切皆是流”,比如我们其实无时不刻都在呼吸对吧,而且吸入和呼出的都是空气,也就是气流,这个很好理解哈~然后我们的嘴巴会去吃东西,就是食物吧,它也会变成食物流,其他的也是一样,像我们看到的外部的风景啊、人物啊、抖音里面的视频啊,都是视觉流,还有我们和其他人之间的谈话、聊天什么的,都是语音流,而且这种视觉流和语音流很多都是持续的,源源不断的。就像这样:

而且这些流就是我们生活中每天都需要面对的各种各样的事情,比如吃了东西,呼吸了新鲜空气,就会让我们获得能量,就不会觉得饿或者没力气了,或者看到外面漂亮的风景,小哥哥小姐姐,以及美食,或者说和朋友谈话、聊天之后就会觉得很愉快,就能引起心理和生理反应,或者说比如和谁一起吃工作餐谈话,谈的不愉快就直接结束,所以这里有个红叉叉,正常结束的话就是一个竖线,比如这段时间对吃的、帅哥美女、聊天或者抖音都没啥兴趣,走冷淡风,就不再对这些外部事件产生反应了,所谓的反应式编程里面所说的反应,其实就是一种对行为的响应。

如果把代码想象成一个人的话,那么那些外部的事情用计算机专业的话来说叫事件,比如用户点击页面按钮发送消息、键盘输入、物联网设备不断地发送信号等等,都是各种输入流,而且程序需要对这些各种事件都做出响应,或者反应,而且要及时。用以前的方式也是可以的,不过如果这种事件很多,涉及的线程、回调就会很多,所以就需要一种新的编程方式来处理这种叫做mashup的混合式Web应用开发。关于反应式编程的一种正式的描述可以看反应式编程官方的《反应式宣言》:

刚才说的只是反应式编程一些概念性或者是感性的认识,真正要了解反应式编程,还是要看代码:

public static void main(String[] args) {

  Person person = new Person("zhangsan", 18);

  // 数据怎么来就怎么处理,处理完后就结束
Observable.just("1", 2, true, 0.618, person, new String("haha"))
.subscribe(System.out::println); }

运行后可以看到,不管是什么样的数据类型,都会被just给展示出来,这就是「流」的概念。而在以前的Java集合List中,肯定是不可能有这种骚操作的。

这种代码示例其实在Reactive的Github官网上面有很多,可以打开它的官网查看:

刚才的代码里面有三个特点:

1、出现了Lambda表达式(System.out::println);

2、出现了流(.just().subscribe());

3、出现了一个叫做Observable的类,而熟悉设计模式的都知道,它是观察者模式中特有的一个类。

下面是观察者模式的一个示意图:

观察者模式也是回调实现的一种方式。观察者模式在开发里面还有另外一个「外号」:发布-订阅模式。经典的观察者模式是像下面这样的:

可以用代码来演示这种剧院和观众的观察者模式:

// 观众的行为(观察者-订阅者接口)
@FunctionalInterface
interface Viewer {
public void watch();
} // 具体订阅者
class ConcreteViewer implements Viewer {
@Override
public void watch() {
System.out.println("正在看表演");
}
}
// 演员(发布者)
class Actor {
// 观众(订阅者)列表
private List<Viewer> viewers = new ArrayList<Viewer>();
// 买票(订阅)
public void buyticket(Viewer viewer) {
viewers.add(viewer);
}
// 退票(取消订阅)
public void refund(Viewer viewer) {
viewers.remove(viewer);
}
// 开始表演
public void play() {
for (Viewer viewer : viewers) {
viewer.watch();
}
}
} /**
* 剧院
*
* @author 湘王
*/
public class Theater {
public static void main(String[] args) {
ConcreteViewer viewer1 = new ConcreteViewer();
ConcreteViewer viewer2 = new ConcreteViewer();
Actor actor = new Actor();
actor.buyticket(viewer1);
actor.buyticket(viewer2);
actor.buyticket(() -> System.out.println("正在VIP包厢看表演"));
actor.play();
}
}

这也是反应式编程的门槛比较高的原因,对于一些没有学过设计模式的同学来说,反应式编程确实很不好理解。这也没办法,因为整个反应式编程的根基就是建立在观察者模式之上的,而且还结合Java9里面的Flow API做了一些改变,有兴趣的同学可以自己去了解一下,所以总结来说的话,反应式编程的特点就是这样的:

用一句简单的话来概括反应式编程 = 函数式编程 + 流式计算 + 观察者模式。


感谢您的大驾光临!咨询技术、产品、运营和管理相关问题,请关注后留言。欢迎骚扰,不胜荣幸~

Java反应式编程(1)的更多相关文章

  1. Java 链式编程

    这里来做一个Java 链式编程的例子,基本就是每次返回一个对象本身,这样就能够去调用对象的方法和属性. package com.sun; public class Demo05 { /** * @pa ...

  2. Java链式编程接口

    在android开发中显示一个AlertDialog时,常采用下列的写法: new AlertDialog.Builder(getApplicationContext()) .setTitle(&qu ...

  3. java链式编程设计

    一般情况下,对一个类的实例和操作,是采用这种方法进行的: Channel channel = new Channel(); channel.queueDeclare(QUEUE_NAME, true, ...

  4. Java反应式框架Reactor中的Mono和Flux

    1. 前言 最近写关于响应式编程的东西有点多,很多同学反映对Flux和Mono这两个Reactor中的概念有点懵逼.但是目前Java响应式编程中我们对这两个对象的接触又最多,诸如Spring WebF ...

  5. Java9第四篇-Reactive Stream API响应式编程

    我计划在后续的一段时间内,写一系列关于java 9的文章,虽然java 9 不像Java 8或者Java 11那样的核心java版本,但是还是有很多的特性值得关注.期待您能关注我,我将把java 9 ...

  6. 使用函数接口和枚举实现配置式编程(Java与Scala实现)

    概述 做报表时,有时需要根据不同的业务生成不同的报表.这样,需要能够动态地配置列字段,并根据列字段来输出对应的报表.使用函数接口结合枚举可以比较优雅地实现配置式编程. 问题描述如下: 假设有对象 St ...

  7. 10-02 Java 形式参数和返回值的问题深入研究,链式编程

    形式参数和返回值的问题: 1:形式参数和返回值的问题(理解) (1)形式参数: 类名:需要该类的对象 抽象类名:需要该类的子类对象 接口名:需要该接口的实现类对象 (2)返回值类型: 类名:返回的是该 ...

  8. Java reactor响应式编程

    转载自:https://www.cnblogs.com/lixinjie/p/a-reactive-streams-on-jvm-is-reactor.html 响应式编程 作为响应式编程方向上的第一 ...

  9. java中的链式编程

    听到链式编程听陌生的,但是写出来就感觉其实很熟悉 package test; public class Test { String name; String phone; String mail; S ...

随机推荐

  1. 第五十一篇:webpack中的loader(二) --less-loader

    好家伙 先扩充一下知识点: 什么是.less文件? 作为一名前端开发的同学,很多时候我们都无法避免地要去写大量的 CSS 代码, 而且耗费的时间还不少,所以学习一种能够提升开发效率的 CSS 预处理器 ...

  2. Docker实用篇

    Docker实用篇 0.学习目标 1.初识Docker 1.1.什么是Docker 微服务虽然具备各种各样的优势,但服务的拆分通用给部署带来了很大的麻烦. 分布式系统中,依赖的组件非常多,不同组件之间 ...

  3. HC32L110(五) Ubuntu20.04 VSCode的Debug环境配置

    目录 HC32L110(一) HC32L110芯片介绍和Win10下的烧录 HC32L110(二) HC32L110在Ubuntu下的烧录 HC32L110(三) HC32L110的GCC工具链和VS ...

  4. KingbaseES例程之快速删除表数据

    概述 快速删除表中的数据 delete语句删除数据 表中的数据被删除了,但是这个数据在硬盘上的真实存储空间不会被释放. 这种删除缺点是:删除效率比较低. 这种删除优点是:支持删除部分数据,支持回滚. ...

  5. KingbaseES R3 集群一键修改集群用户密码案例

    案例说明: 在KingbaseES R3集群的最新版本中增加了kingbase_monitor.sh一键修改集群用户密码的功能,本案例是对此功能的测试. kingbaseES R3集群一键修改密码说明 ...

  6. 弱隔离级别 & 事务并发问题

    介绍弱隔离级别 为什么要有弱隔离级别 如果两个事务操作的是不同的数据, 即不存在数据依赖关系, 则它们可以安全地并行执行.但是当出现某个事务修改数据而另一个事务同时要读取该数据, 或者两个事务同时修改 ...

  7. MySQL建表语句生成Golang代码

    1. 背景 对于后台开发新的需求时,一般会先进行各种表的设计,写各个表的建表语句 然后根据建立的表,写对应的model代码.基础的增删改查代码(基础的增删改查服务可以划入DAO(Data Access ...

  8. 选择排序C语言版本

    算法思路,从头至尾扫描序列. 首先从第二个到最后,找出最小的一个元素,和第一个元素交换: 接着从第三个到最后,后面找出最小的一个元素,和第二个元素交换: 依次类推最终得到一个有序序列. void Se ...

  9. AVL Tree (1) - Definition, find and Rotation

    1. 定义 (15-1) [AVL tree]: 一棵空二叉树是 AVL tree; 若 T 是一棵非空二叉树, 则 T 满足以下两个条件时, T 是一棵 AVL tree: T_LeftSubtre ...

  10. 发布日志 - kratos v2.1.0 版本发布

    github https://github.com/go-kratos/kratos/releases/tag/v2.1.0 新的功能 新增客户端负载均衡器(load balancing)和路由选择器 ...