ChatGPT解决这个技术问题 Extra ChatGPT

什么是(函数式)反应式编程?

锁定。这个问题及其答案被锁定,因为这个问题离题但具有历史意义。它目前不接受新的答案或交互。

我已阅读有关 reactive programming 的 Wikipedia 文章。我还阅读了关于 functional reactive programming 的小文章。描述很抽象。

函数式反应式编程(FRP)在实践中意味着什么?反应式编程(相对于非反应式编程?)由什么组成?

我的背景是命令式/OO 语言,因此将不胜感激与此范例相关的解释。

这是一个具有活跃想象力和良好讲故事技巧的人,可以处理整个事情。 paulstovell.com/reactive-programming
有人真的需要为我们这里的自学者写一个“傻瓜函数式反应式编程”。我找到的每一个资源,甚至 Elm,似乎都假设你在过去五年中获得了 CS 硕士学位。那些了解 FRP 的人似乎完全失去了从幼稚的角度看待问题的能力,这对教学、培训和传福音至关重要。
另一个出色的 FRP 介绍:The introduction to Reactive Programming you've been missing 我的同事 André
我见过的最好的之一,基于示例:gist.github.com/staltz/868e7e9bc2a7b8c1f754
我发现电子表格类比作为第一印象非常有用(参见 Bob 的回答:stackoverflow.com/a/1033066/1593924)。电子表格单元格对其他单元格中的更改做出反应(拉动),但不会伸出并更改其他单元格(不推送)。最终结果是您可以更改一个单元格,而无数其他单元格“独立”更新自己的显示。

C
Conal

如果您想了解 FRP,可以从 1998 年的旧 Fran tutorial 开始,它有动画插图。对于论文,从 Functional Reactive Animation 开始,然后跟进我主页上的出版物链接和 Haskell wiki 上的 FRP 链接。

就个人而言,我喜欢在讨论 FRP 的实施方式之前先考虑一下 FRP 的含义。 (没有规范的代码是没有问题的答案,因此“甚至没有错”。)所以我不会像 Thomas K 在另一个答案(图、节点、边、触发、执行、 ETC)。有许多可能的实现样式,但没有一个实现说明 FRP 是什么。

我确实赞同 Laurence G 的简单描述,即 FRP 是关于“表示‘随时间’的值的数据类型”。传统的命令式编程仅通过状态和突变间接地捕获这些动态值。完整的历史(过去、现在、未来)没有一流的表现。此外,只能(间接)捕获离散演变的值,因为命令式范式在时间上是离散的。相比之下,FRP 直接捕获这些不断变化的值,并且在不断变化的值方面没有任何困难。

FRP 的不寻常之处还在于它是并发的,而不会与困扰命令式并发的理论和实用鼠窝发生冲突。从语义上讲,FRP 的并发是细粒度的、确定的、连续的。 (我说的是意义,而不是实现。实现可能涉及也可能不涉及并发或并行性。)语义确定性对于推理非常重要,无论是严格的还是非正式的。虽然并发给命令式编程增加了巨大的复杂性(由于非确定性交错),但在 FRP 中却毫不费力。

那么,什么是玻璃钢?你本可以自己发明的。从这些想法开始:

动态/不断变化的价值(即“随时间变化”的价值)本身就是一流的价值。您可以定义它们并将它们组合起来,将它们传入和传出函数。我把这些东西称为“行为”。

行为由几个原语构成,例如恒定(静态)行为和时间(如时钟),然后是顺序和并行组合。 n 行为通过应用 n 元函数(静态值)“逐点”组合,即随着时间的推移连续。

为了解释离散现象,有另一种类型(家族)的“事件”,每个事件都有一个(有限或无限)事件流。每个事件都有一个关联的时间和值。

要提出可以构建所有行为和事件的组合词汇,请尝试一些示例。继续解构为更一般/更简单的部分。

为了让你知道你在坚实的基础上,给整个模型一个组合基础,使用指称语义技术,这意味着(a)每种类型都有一个相应的简单而精确的数学类型的“意义”,并且( b) 每个原语和操作符都有一个简单而精确的含义,作为成分含义的函数。永远不要将实施考虑因素混入您的探索过程中。如果此描述对您来说是胡言乱语,请查阅 (a) 具有类型类态射的指称设计,(b) 推拉函数式反应式编程(忽略实现位),以及 (c) 指称语义 Haskell wikibooks 页面。请注意,指称语义有两个部分,来自其两位创始人 Christopher Strachey 和 Dana Scott:更容易且更有用的 Strachey 部分和更难且不太有用(用于软件设计)的 Scott 部分。

如果你坚持这些原则,我希望你会或多或少地获得 FRP 精神的东西。

我从哪里得到这些原则?在软件设计中,我总是问同样的问题:“这是什么意思?”。指称语义为我提供了一个精确的框架来解决这个问题,并且符合我的审美(与操作语义或公理语义不同,这两者都让我不满意)。所以我问自己什么是行为?我很快意识到命令式计算的时间离散性是对特定类型机器的适应,而不是对行为本身的自然描述。我能想到的对行为的最简单精确描述就是“(连续)时间的函数”,这就是我的模型。令人高兴的是,这个模型可以轻松优雅地处理连续的、确定性的并发。

正确有效地实施这个模型是一个相当大的挑战,但那是另一回事了。


我一直知道函数式反应式编程。它似乎与我自己的研究(在交互式统计图形中)有关,我相信其中的许多想法都会对我的工作有所帮助。然而,我发现很难通过这种语言——我必须真正了解“指称语义”和“类型类态射”才能理解发生了什么吗?对该主题的一般观众介绍将非常有用。
@Conal:您清楚地知道自己在说什么,但是您的语言假定我拥有计算数学博士学位,而我没有。我确实有系统工程背景和 20 多年的计算机和编程语言经验,但我仍然觉得你的回答让我感到困惑。我挑战你用英文重新发布你的回复;-)
@minplay.dk:你的言论并没有给我太多关于你不理解的内容,而且我不愿意对你正在寻找的特定英语子集做出疯狂的猜测。但是,我邀请您具体说明我上面的解释的哪些方面您遇到了绊脚石,以便我和其他人可以帮助您。例如,是否有您想要定义的特定词或您想要添加参考的概念?我真的很喜欢提高我写作的清晰度和可访问性——而不是让它变得愚蠢。
“确定性”/“确定”意味着有一个单一的、明确定义的正确值。相比之下,几乎所有形式的命令式并发都可以给出不同的答案,这取决于调度程序或您是否正在查看,它们甚至会死锁。 “语义”(更具体地说是“指称”)是指表达式或表示的值(“指称”),与“操作”(如何计算答案或由什么消耗多少空间和/或时间)形成对比一种机器)。
我同意@mindplay.dk,尽管我不能吹嘘自己已经在这个领域工作了很长时间。即使您似乎知道自己在说什么,但它并没有让我快速、简短和简单地理解这是什么,因为我已经被宠坏了,可以期待 SO。这个答案主要是让我在没有真正回答我的第一个问题的情况下提出大量新问题。我希望分享在该领域仍然相对无知的经验可以让您了解您真正需要的简单和简短。顺便说一句,我来自与 OP 相似的背景。
L
Laurence Gonsalves

在纯函数式编程中,没有副作用。对于许多类型的软件(例如,任何与用户交互的软件),副作用在某种程度上是必要的。

在保持函数式风格的同时获得类似副作用的行为的一种方法是使用函数式反应式编程。这是函数式编程和反应式编程的结合。 (您链接到的维基百科文章是关于后者的。)

反应式编程背后的基本思想是,某些数据类型代表“随着时间的推移”的值。涉及这些随时间变化的值的计算本身将具有随时间变化的值。

例如,您可以将鼠标坐标表示为一对随时间变化的整数值。假设我们有类似的东西(这是伪代码):

x = <mouse-x>;
y = <mouse-y>;

在任何时候,x 和 y 都会有鼠标的坐标。与非反应式编程不同,我们只需要进行一次赋值,x 和 y 变量将自动保持“最新”。这就是为什么反应式编程和函数式编程可以很好地结合在一起的原因:反应式编程消除了对变量进行变异的需要,同时仍然让您可以做很多可以通过变量变异完成的事情。

如果我们然后基于此进行一些计算,则结果值也将是随时间变化的值。例如:

minX = x - 16;
minY = y - 16;
maxX = x + 16;
maxY = y + 16;

在此示例中,minX 将始终比鼠标指针的 x 坐标小 16。使用响应式库,您可以这样说:

rectangle(minX, minY, maxX, maxY)

并且将围绕鼠标指针绘制一个 32x32 的框,并将跟踪它移动到哪里。

这是一个很好的paper on functional reactive programming


那么反应式编程是声明式编程的一种形式吗?
> 那么反应式编程是声明式编程的一种形式吗?函数式反应式编程是函数式编程的一种形式,它是声明式编程的一种形式。
@user712092 不是真的,不。例如,如果我在 C 中使用您的宏调用 sqrt(x),它只会计算 sqrt(mouse_x()) 并返回一个双精度值。在真正的函数式反应系统中,sqrt(x) 将返回一个新的“随时间加倍”。如果您要尝试使用 #define 模拟 FR 系统,您几乎必须放弃变量以支持宏。 FR 系统通常也只会在需要重新计算时才重新计算内容,而使用宏意味着您将不断地重新评估所有内容,一直到子表达式。
“对于许多类型的软件(例如,任何与用户交互的软件),副作用在某种程度上是必要的。”而且也许只在实施层面。纯粹的惰性函数式编程的实现有很多副作用,而范式的成功之一就是将许多这些影响排除在编程模型之外。我自己对功能性用户界面的尝试表明,它们也可以完全编程而没有副作用。
@tieTYT x 永远不会重新分配/变异。 x 的值是随时间变化的值序列。另一种看待它的方式是,x 的值不是像数字那样具有“正常”值,而是(从概念上)是一个将时间作为参数的函数。 (这有点过于简单化了。您无法创建时间值来预测鼠标位置等事物的未来。)
佚名

获得关于它是什么样的第一直觉的一种简单方法是想象你的程序是一个电子表格,你的所有变量都是单元格。如果电子表格中的任何单元格发生更改,则引用该单元格的任何单元格也会更改。玻璃钢也是如此。现在想象一些单元格自己改变(或者更确切地说,是从外部世界获取的):在 GUI 情况下,鼠标的位置就是一个很好的例子。

这必然会错过很多。当您实际使用 FRP 系统时,这个比喻很快就失效了。一方面,通常也会尝试对离散事件进行建模(例如,单击鼠标)。我把它放在这里只是为了让你知道它是什么样的。


一个极其贴切的例子。有理论的东西很好,也许有些人在不求助于一个基础示例的情况下得到了它的含义,但我需要从它对我的作用开始,而不是它抽象地是什么。我最近才得到的(来自 Netflix 的 Rx 演讲!)是 RP(或 Rx,无论如何),使这些“变化的值”成为头等舱,让你对它们进行推理,或者编写与它们一起做事的函数。如果您愿意,可以编写函数来创建电子表格或单元格。它处理值何时结束(消失)并让您自动清理。
此示例强调了事件驱动编程和反应式方法之间的区别,您只需声明依赖项以使用智能路由。
G
Gilles 'SO- stop being evil'

对我来说,符号 = 有两种不同的含义:

在数学中 x = sin(t) 意味着 x 是 sin(t) 的不同名称。所以写 x + y 和 sin(t) + y 是一样的。函数式反应式编程在这方面就像数学:如果你写 x + y,它是用 t 在使用时的值来计算的。在类 C 编程语言(命令式语言)中,x = sin(t) 是一个赋值:这意味着 x 存储在赋值时取的 sin(t) 的值。


很好的解释。我认为您还可以添加 FRP 意义上的“时间”通常是“来自外部输入的任何变化”。每当外力改变 FRP 的输入时,您已经将“时间”向前移动,并重新计算受更改影响的所有内容。
在数学中,x = sin(t) 意味着 x 是给定 tsin(t) 的值。它不是作为函数的 sin(t) 的不同名称。否则它将是 x(t) = sin(t)
+Dmitri Zaitsev 等号在数学中有多种含义。其中之一是每当您看到左侧时,您都可以将其与右侧交换。例如 2 + 3 = 5a**2 + b**2 = c**2
T
Thomas Kammeyer

好的,从背景知识和阅读您指出的维基百科页面来看,反应式编程似乎类似于数据流计算,但具有特定的外部“刺激”,触发一组节点触发并执行它们的计算。

这非常适合 UI 设计,例如,触摸用户界面控件(例如,音乐播放应用程序上的音量控件)可能需要更新各种显示项目和音频输出的实际音量。当您修改与修改与有向图中的节点关联的值相对应的音量(比方说滑块)时。

具有来自该“体积值”节点的边缘的各种节点将被自动触发,并且任何必要的计算和更新都会自然地在应用程序中波动。应用程序对用户刺激做出“反应”。函数式反应式编程只是用函数式语言或通常在函数式编程范式中实现这个想法。

有关“数据流计算”的更多信息,请在 Wikipedia 上搜索这两个词或使用您最喜欢的搜索引擎。总体思路是这样的:程序是节点的有向图,每个节点执行一些简单的计算。这些节点通过图形链接相互连接,这些链接将一些节点的输出提供给其他节点的输入。

当一个节点触发或执行它的计算时,连接到它的输出的节点有它们相应的输入“触发”或“标记”。任何具有所有输入触发/标记/可用的节点都会自动触发。该图可能是隐式的或显式的,具体取决于反应式编程的实现方式。

可以将节点视为并行触发,但它们通常是串行执行的或以有限的并行性执行(例如,可能有几个线程在执行它们)。一个著名的例子是 Manchester Dataflow Machine,它 (IIRC) 使用标记数据架构通过一个或多个执行单元来调度图中节点的执行。数据流计算非常适合异步触发计算以产生级联计算的情况,而不是试图让执行由一个(或多个时钟)控制。

反应式编程引入了这种“级联执行”的思想,并且似乎以类似数据流的方式来考虑程序,但前提是一些节点与“外部世界”挂钩,并且当这些感觉时触发级联执行-like节点改变。程序执行看起来就像一个复杂的反射弧。该程序可能在刺激之间基本上是固着的,也可能不是基本上固着的,或者可能在刺激之间进入基本上固着的状态。

“非反应性”编程将是对执行流程和与外部输入的关系有非常不同的看法的编程。这可能有点主观,因为人们可能会倾向于说出任何对外部输入做出“反应”的东西。但是从本质上看,以固定间隔轮询事件队列并将发现的任何事件分派给函数(或线程)的程序反应性较小(因为它只以固定间隔处理用户输入)。再说一次,这就是这里的精神:人们可以想象将一个具有快速轮询间隔的轮询实现放在一个非常低级别的系统中,并在其之上以一种反应方式进行编程。


好的,现在上面有一些很好的答案。我应该删除我的帖子吗?如果我看到两三个人说它什么也没增加,我会删除它,除非它的有用数量增加。除非它增加了一些价值,否则将它留在这里是没有意义的。
您提到了数据流,因此恕我直言,这增加了一些价值。
这就是 QML 的本意,似乎 ;)
对我来说,这个答案是最容易理解的,尤其是因为使用了“通过应用程序产生涟漪”和“类似感官的节点”等自然类似物。伟大的!
不幸的是,曼彻斯特数据流机器链接已失效。
j
jhegedus

在阅读了很多关于 FRP 的页面后,我终于看到了 this 关于 FRP 的启发性文章,它终于让我明白了 FRP 的真正含义。

我在下面引用 Heinrich Apfelmus(反应香蕉的作者)。

函数式反应式编程的本质是什么?一个常见的答案是“FRP 就是用时变函数而不是可变状态来描述系统”,这当然不会错。这是语义的观点。但在我看来,更深层次、更令人满意的答案是由以下纯语法标准给出的:函数式反应式编程的本质是在声明时完全指定值的动态行为。例如,以计数器为例:您有两个标记为“向上”和“向下”的按钮,可用于增加或减少计数器。必须先指定一个初始值,然后在按下按钮时更改它;像这样: counter := 0 -- buttonUp 上的初始值 = (counter := counter + 1) -- 稍后在 buttonDown 上更改它 = (counter := counter - 1) 关键是在声明时,只有指定计数器的初始值;计数器的动态行为隐含在程序文本的其余部分中。相比之下,函数式反应式编程在声明时指定了整个动态行为,如下所示: counter :: Behavior Int counter = accumulate ($) 0 (fmap (+1) eventUp `union` fmap (subtract 1) eventDown) 每当你想了解计数器的动态,你只需要看看它的定义。它可能发生的一切都会出现在右侧。这与后续声明可以改变先前声明值的动态行为的命令式方法形成鲜明对比。

https://i.stack.imgur.com/OyHY7.jpg

j 是离散的:1,2,3,4...

f 取决于 t,因此这包含了对外部刺激进行建模的可能性

程序的所有状态都封装在变量 x_i

FRP 库负责处理时间进度,换句话说,将 j 带到 j+1

我在 this 视频中更详细地解释了这些方程式。

编辑:

在原始答案之后大约 2 年,最近我得出结论,FRP 实施还有另一个重要方面。他们需要(并且通常会)解决一个重要的实际问题:缓存失效。

x_i-s 的方程描述了一个依赖图。当某些 x_i 在时间 j 发生变化时,不需要更新 j+1 处的所有其他 x_i' 值,因此不需要重新计算所有依赖项,因为某些 x_i' 可能独立于 { 1}。

此外,发生变化的 x_i-s 可以增量更新。例如,让我们考虑 Scala 中的映射操作 f=g.map(_+1),其中 fgIntsList。这里 f 对应于 x_i(t_j),而 gx_j(t_j)。现在,如果我在 g 前添加一个元素,那么对 g 中的所有元素执行 map 操作会很浪费。一些 FRP 实现(例如 reflex-frp)旨在解决这个问题。此问题也称为 incremental computing.

换句话说,FRP 中的行为(x_i-s)可以被认为是缓存计算。如果某些 f_i-s 发生更改,则 FRP 引擎的任务是有效地使这些缓存-s(x_i-s)无效并重新计算。


在你使用离散方程之前,我一直在你身边。 FRP的创始思想是连续时间,这里没有“j+1”。相反,想想连续时间的函数。正如牛顿、莱布尼茨和其他人向我们展示的那样,使用积分和 ODE 系统以微分但连续的方式描述这些函数通常非常方便(并且从字面意义上“自然”)。否则,您描述的是一种近似算法(而且是一个糟糕的算法),而不是事物本身。
HTML 模板和布局约束语言 layx 似乎表达了 FRP 的元素。
@Conal 这让我想知道 FRP 与 ODE 有何不同。它们有何不同?
@jhegedus 在该集成中(可能是递归的,即ODE)提供了FRP 的构建块之一,而不是全部。 FRP 词汇表的每一个元素(包括但不限于积分)都用连续时间精确解释。这个解释有用吗?
P
Peter Mortensen

Conal Elliott 的论文Simply efficient functional reactivity (direct PDF, 233 KB) 是一个相当不错的介绍。相应的库也可以工作。

该论文现在被另一篇论文 Push-pull functional reactive programming (direct PDF, 286 KB) 取代。


t
tldr

免责声明:我的答案是在 rx.js 的上下文中 - 一个用于 Javascript 的“反应式编程”库。

在函数式编程中,不是遍历集合的每个项目,而是将高阶函数 (HoF) 应用于集合本身。因此,FRP 背后的想法是,与其处理每个单独的事件,不如创建一个事件流(使用 observable* 实现)并将 HoF 应用于该事件。通过这种方式,您可以将系统可视化为连接发布者和订阅者的数据管道。

使用 observable 的主要优点是:i) 它从您的代码中抽象出状态,例如,如果您希望事件处理程序仅在每个“n”个事件时触发,或者在第一个“n”个事件后停止触发,或仅在第一个“n”事件后开始触发,您可以只使用 HoF(分别为 filter、takeUntil、skip)而不是设置、更新和检查计数器。 ii) 它提高了代码局部性——如果你有 5 个不同的事件处理程序来改变组件的状态,你可以合并它们的 observables 并在合并的 observable 上定义一个事件处理程序,从而有效地将 5 个事件处理程序合并为 1。这使得它非常很容易推断整个系统中的哪些事件会影响组件,因为它们都存在于单个处理程序中。

Observable 是 Iterable 的对偶。

Iterable 是一个惰性消耗的序列 - 每个项目在它想要使用它时由迭代器拉取,因此枚举由消费者驱动。

一个可观察对象是一个延迟生成的序列——每一项被添加到序列中时都会被推送给观察者,因此枚举是由生产者驱动的。


非常感谢您对可观察对象的直接定义及其与可迭代对象的区别。我认为将一个复杂的概念与其众所周知的对偶概念进行比较通常非常有助于获得真正的理解。
“因此,FRP 背后的想法是,不是处理每个单独的事件,而是创建一个事件流(使用 observable* 实现)并将 HoF 应用于该事件。” 我可能弄错了,但我相信这一点实际上不是 FRP,而是对观察者设计模式的一个很好的抽象,它允许通过 HoF 进行功能操作(这很棒!),同时仍然打算与命令式代码一起使用。关于主题的讨论 - lambda-the-ultimate.org/node/4982
D
Dan Ross

伙计,这真是一个绝妙的主意!为什么我在 1998 年没有发现这件事?无论如何,这是我对 Fran 教程的解释。非常欢迎提出建议,我正在考虑基于此启动游戏引擎。

import pygame
from pygame.surface import Surface
from pygame.sprite import Sprite, Group
from pygame.locals import *
from time import time as epoch_delta
from math import sin, pi
from copy import copy

pygame.init()
screen = pygame.display.set_mode((600,400))
pygame.display.set_caption('Functional Reactive System Demo')

class Time:
    def __float__(self):
        return epoch_delta()
time = Time()

class Function:
    def __init__(self, var, func, phase = 0., scale = 1., offset = 0.):
        self.var = var
        self.func = func
        self.phase = phase
        self.scale = scale
        self.offset = offset
    def copy(self):
        return copy(self)
    def __float__(self):
        return self.func(float(self.var) + float(self.phase)) * float(self.scale) + float(self.offset)
    def __int__(self):
        return int(float(self))
    def __add__(self, n):
        result = self.copy()
        result.offset += n
        return result
    def __mul__(self, n):
        result = self.copy()
        result.scale += n
        return result
    def __inv__(self):
        result = self.copy()
        result.scale *= -1.
        return result
    def __abs__(self):
        return Function(self, abs)

def FuncTime(func, phase = 0., scale = 1., offset = 0.):
    global time
    return Function(time, func, phase, scale, offset)

def SinTime(phase = 0., scale = 1., offset = 0.):
    return FuncTime(sin, phase, scale, offset)
sin_time = SinTime()

def CosTime(phase = 0., scale = 1., offset = 0.):
    phase += pi / 2.
    return SinTime(phase, scale, offset)
cos_time = CosTime()

class Circle:
    def __init__(self, x, y, radius):
        self.x = x
        self.y = y
        self.radius = radius
    @property
    def size(self):
        return [self.radius * 2] * 2
circle = Circle(
        x = cos_time * 200 + 250,
        y = abs(sin_time) * 200 + 50,
        radius = 50)

class CircleView(Sprite):
    def __init__(self, model, color = (255, 0, 0)):
        Sprite.__init__(self)
        self.color = color
        self.model = model
        self.image = Surface([model.radius * 2] * 2).convert_alpha()
        self.rect = self.image.get_rect()
        pygame.draw.ellipse(self.image, self.color, self.rect)
    def update(self):
        self.rect[:] = int(self.model.x), int(self.model.y), self.model.radius * 2, self.model.radius * 2
circle_view = CircleView(circle)

sprites = Group(circle_view)
running = True
while running:
    for event in pygame.event.get():
        if event.type == QUIT:
            running = False
        if event.type == KEYDOWN and event.key == K_ESCAPE:
            running = False
    screen.fill((0, 0, 0))
    sprites.update()
    sprites.draw(screen)
    pygame.display.flip()
pygame.quit()

简而言之:如果每个组件都可以被视为一个数字,那么整个系统就可以被视为一个数学方程式,对吧?


这有点晚了,但无论如何... Frag is a game using FRP
D
David Lemon

Paul Hudak 的书 The Haskell School of Expression 不仅很好地介绍了 Haskell,而且在 FRP 上也花费了相当多的时间。如果您是 FRP 的初学者,我强烈推荐它让您了解 FRP 的工作原理。

还有看起来像是对这本书的新重写(2011 年发布,2014 年更新),The Haskell School of Music


Y
Yuning

根据前面的答案,似乎在数学上,我们只是以更高的顺序思考。我们不考虑类型 X 的值 x,而是考虑函数 x:T → X,其中 T 是时间的类型,可以是自然数、整数或连续统。现在,当我们在编程语言中写 y := x + 1 时,我们实际上是指方程 y(t) = x(t) + 1。


e
emperorz

如前所述,就像电子表格一样。通常基于事件驱动的框架。

与所有“范式”一样,它的新颖性值得商榷。

根据我对参与者分布式流网络的经验,它很容易成为节点网络中状态一致性的一般问题的牺牲品,即您最终会陷入大量的振荡和奇怪的循环中。

这很难避免,因为某些语义意味着引用循环或广播,并且当参与者网络在某些不可预测的状态上收敛(或不收敛)时可能会非常混乱。

类似地,尽管具有明确定义的边缘,但可能无法达到某些状态,因为全局状态会远离解决方案。 2+2 可能会也可能不会成为 4,这取决于 2 何时变为 2,以及他们是否保持这种状态。电子表格具有同步时钟和环路检测。分布式演员通常不会。

一切都很有趣:)。


D
Daniel Kaplan

我在 Clojure subreddit 上找到了这个关于 FRP 的精彩视频。即使您不了解 Clojure,也很容易理解。

这是视频:http://www.youtube.com/watch?v=nket0K1RXU4

以下是视频在后半部分引用的来源:https://github.com/Cicayda/yolk-examples/blob/master/src/yolk_examples/client/autocomplete.cljs


C
Community

Andre Staltz 的 This article 是迄今为止我所见过的最好和最清晰的解释。

文章中的一些引用:

反应式编程是使用异步数据流进行编程。最重要的是,您将获得一个惊人的功能工具箱来组合、创建和过滤任何这些流。

以下是文章中精彩图表的示例:

https://i.stack.imgur.com/OTya4.png


D
Dmitri Zaitsev

它是关于随时间(或忽略时间)的数学数据转换。

在代码中,这意味着功能纯度和声明式编程。

状态错误是标准命令式范式中的一个大问题。不同的代码位可能会在程序执行的不同“时间”改变一些共享状态。这很难处理。

在 FRP 中,您描述(如在声明式编程中)数据如何从一种状态转换为另一种状态以及触发它的原因。这允许您忽略时间,因为您的函数只是对其输入做出反应并使用它们的当前值来创建一个新值。这意味着状态包含在转换节点的图(或树)中,并且在功能上是纯的。

这大大降低了复杂性和调试时间。

想想数学中的 A=B+C 和程序中的 A=B+C 之间的区别。在数学中,您正在描述一种永远不会改变的关系。在一个程序中,它说“现在”A 是 B+C。但是下一个命令可能是 B++,在这种情况下 A 不等于 B+C。在数学或声明式编程中,无论您询问什么时间点,A 总是等于 B+C。

因此,通过消除共享状态的复杂性并随时间改变值。你的程序更容易推理。

EventStream 是 EventStream + 一些转换函数。

行为是 EventStream + 内存中的某个值。

当事件触发时,通过运行转换函数更新值。这产生的值存储在行为内存中。

可以组合行为以产生新的行为,这些行为是对其他 N 个行为的转换。这个组合值将在输入事件(行为)触发时重新计算。

“由于观察者是无状态的,我们经常需要其中的几个来模拟一个状态机,就像在拖动示例中一样。我们必须将状态保存在所有相关观察者都可以访问的地方,例如上面的变量路径中。”

引自 - 弃用观察者模式 http://infoscience.epfl.ch/record/148043/files/DeprecatingObserversTR2010.pdf


这正是我对声明式编程的感受,你只是比我更好地描述了这个想法。
p
pdorgambide

关于响应式编程的简短而清晰的解释出现在 Cyclejs - Reactive Programming 上,它使用简单直观的示例。

[模块/组件/对象] 是反应式的,意味着它完全负责通过对外部事件做出反应来管理自己的状态。这种方法有什么好处?它是控制反转,主要是因为[模块/组件/对象]对自己负责,使用私有方法对公共方法改进封装。

这是一个很好的起点,而不是完整的知识来源。从那里你可以跳到更复杂和更深入的论文。


S
Sentinel

查看 Rx,.NET 的反应式扩展。他们指出,使用 IEnumerable,您基本上是从流中“拉取”。在 IQueryable/IEnumerable 上的 Linq 查询是从集合中“吸出”结果的集合操作。但是在 IObservable 上使用相同的运算符,您可以编写“反应”的 Linq 查询。

例如,您可以编写一个 Linq 查询,例如 (from m in MyObservableSetOfMouseMovements where mX<100 and mY<100 select new Point(mX,mY))。

有了 Rx 扩展,就是这样:你的 UI 代码可以对传入的鼠标移动流做出反应,并在你处于 100,100 框时进行绘制......


K
Krishna Ganeriwal

FRP 是函数式编程(基于一切都是函数的思想的编程范式)和反应式编程范式(基于一切都是流的思想(观察者和可观察的哲学))的组合。它应该是世界上最好的。

查看关于反应式编程的 Andre Staltz 帖子。