在最近几个主要版本的Java的每次迭代中,似乎都有一致的新方法来管理并发任务。
在Java中9,我们有流API这类似于可流动API
RxJava,但与Java 9有一个更简单的一组类和接口。
Java 9
有Flow.Publisher
,Flow.Subscriber
,Flow.Processor
,Flow.Subscription
,和SubmissionPublisher
,这就是它。
RxJava的
拥有全 包
的流API状类,即io.reactivex.flowables
,io.reactivex.subscribers
,io.reactivex.processors
,io.reactivex.observers
,和io.reactivex.observables
这似乎做同样的事情。
这两个库之间的主要区别是什么?为什么有人会使用Java 9 Flow库而不是种类繁多的RxJava库,反之亦然?
这两个库之间的主要区别是什么?
Java 9 Flow API不是独立的库,而是Java Standard Edition库的组件,由从2015年初建立的Reactive
Streams
规范采用的4个接口组成。从理论上讲,它的包含可以实现JDK中的特定用法,例如孵化中的HttpClient,当然也可能是计划中的Async
Database Connection SubmissionPublisher
。
RxJava是Java库,它使用ReactiveX样式的API设计来为响应(推送)数据流提供丰富的运算符集。版本2
Flowable
以及其他版本XxxProcessor
,实现了Reactive Streams
API,该API允许Flowable
其他兼容库使用实例,然后又可以将其中的任何实例包装Publisher
为,Flowable
以使用这些实例,并与它们组合丰富的运算符。
因此,Reactive Streams API是 最小的接口规范, 而RxJava 2是它的一个 实现
,再加上RxJava声明了大量的其他方法,以形成自己丰富而流畅的API。
RxJava 1在其他来源中启发了Reactive Streams规范,但无法利用它(必须保持兼容)。RxJava
2是完整的重写和单独的主要版本,可以包含和使用Reactive
Streams规范(由于Rsc项目,甚至可以在内部对其进行扩展),并且已经比Java 9早一年发布了。决定v1和v2都继续支持Java
6,因此支持许多Android运行时。因此,它不能直接利用Java 9现在提供的Flow
API,而只能通过桥接器。其他基于Reactive
Streams的库也需要和/或提供这种桥接。
RxJava 3可能针对Java 9 Flow
API,但这尚未确定,并且取决于后续Java版本带来的功能(即值类型),我们可能会在一年左右的时间内没有v3。
直到那时,还有一个名为Reactive4JavaFlow的原型库,它确实实现了Flow
API,并在其上提供了ReactiveX样式丰富的流畅API。
为什么有人会使用Java 9 Flow库而不是种类繁多的RxJava库,反之亦然?
Flow API是互操作规范,而不是最终用户API。通常,您不会直接使用它,而是将流传递给它的各种实现。在讨论JEP
266时,作者没有发现任何现有库的API足以使Flow API具有默认值(不同于rich
java.util.Stream
)。因此,决定用户现在必须依靠第三方实现。
您必须等待现有的反应式库通过它们自己的桥实现或要实现的新库来本地支持Flow API。
在Flow API上提供丰富的运算符集只是库要实现它的原因。数据源供应商(例如,反应性数据库驱动程序,网络库)可以开始通过Flow
API实现其自己的数据访问器,并依靠丰富的库来包装它们,并为它们提供转换和协调,而不必强迫所有人实施所有这些运算符。
因此,更好的问题是,您现在应该立即使用基于Flow API的互操作还是坚持使用反应式流?
如果您需要相对较快的工作时间和可靠的解决方案,建议您暂时坚持使用Reactive
Streams生态系统。如果您有足够的时间或想要探索事物,则可以开始使用Flow API。
问题内容: 我错放了太多次了,我想我一直忘记,因为我不知道两者之间的区别,只是一个给了我我期望的价值,而另一个却没有。 为什么是这样? 问题答案: 是的简写形式(尽管请注意,该表达式只会被计算一次。) 是的,即指定一元的到。 例子:
问题内容: 因此,我有一段简单的代码可以打印出整数1-10: 然后,如果仅在第3行上更改一个运算符,它将打印出无限数量的1整数(我知道为什么会这样做)。为什么在运行第二个程序时没有出现语法错误?如果赋值运算符后面跟着一个加法运算符,它不会调用语法错误吗? 问题答案: 与相同, 只是意味着。
问题内容: 有人可以解释一下 和 我不知道“确切”的含义 问题答案: 在这个例子中,什么都没有。当您具有多个具有相似名称的路径时,该参数将起作用: 例如,假设我们有一个显示用户列表的组件。我们还有一个用于创建用户的组件。的网址应嵌套在下。因此,我们的设置可能如下所示: 现在,这里的问题是,当我们转到路由器时,将通过所有定义的路由,并返回它找到的第一个匹配项。因此,在这种情况下,它将首先找到路线,然
问题内容: 我很好奇printStackTrace()和toString()之间的区别是什么。乍一看,他们 似乎 做的完全相同。 码: 问题答案: 不,有重要区别!使用toString,您只有异常的类型和错误消息。使用printStackTrace()可以获得异常的整个堆栈跟踪,这对于调试非常有帮助。 System.out.println(toString())的示例: printStackTra
问题内容: 我看不到两种方式之间的任何区别,@ Qualifier 始终与 @Autowired一起使用 。 VS 有人能让我知道其中的区别吗?谢谢! 问题答案: 可以单独使用。如果单独使用,将按类型进行接线。因此,如果在容器中声明了多个相同类型的bean,而又不知道要注入哪个bean,就会出现问题。结果,通过指定Bean名称(按名称进行绑定),与一起使用来阐明要实际连接的Bean 也按名称接线。
问题内容: System.getenv()和System.getProperty()有什么区别? 当我们使用Processbuilder运行任何命令时,我们可以设置环境变量,即: 如果在此方法之前使用System.setProperties()设置了一些属性,则该属性可用于由ProcessBuilder启动的此过程吗? 问题答案: 获取 环境变量 。获取Java属性。环境变量是在OS级别指定的。通
问题内容: 和以下代码之间有什么区别: 和 他们是同义词吗?在某些情况下,一个比另一个更可取吗?使用这两种方法时该做什么和不该做什么? 问题答案: Class.forName()将始终使用调用者的ClassLoader,而ClassLoader.loadClass()可以指定其他ClassLoader。我相信Class.forName也会初始化加载的类,而ClassLoader.loadClass