前言

我以为

第一次接触 swift 语言时,看到函数的表示形式如下:

func fun(num: Int) -> Int {
return num + 1
} let f = fun(1)

和Objective-C对比一下:

- (int)fun:(int)num
{
return num + 1;
} int f = [self fun:1];

我们很容易就认为 -> 和Objective-C里面的 - 一样嘛,一种修饰符而已,-> Int- (int)就是一个东西嘛!只不过 swift 在后面(和Golang的函数定义蛮像的,Golang没有这个->而已。),OC在前而已,新语言嘛,要和古老的 OC 有所区别的嘛。

() -> ()

回想一下C语言的函数指针定义,可以定义一个函数指针为:

typedef int (*funPointer)(int); // funPointer = int (*)(int)

如果按此举一反三的话,那么 swift 版的 fun 就可以看成这样:

fun = (Int) -> Int

把参数什么的都去掉,变成如下:

fun = () -> ()

() -> () 就是我们今天讨论的重点了。() 表示数据,->表示运算过程,fun函数就是从输入一个数据,到得到一个数据的过程, 即值的变化过程。

(sunnyxx的推导过程:《() -> ()》)

()

() -> () 表示值的变化的过程,是函数式编程范式的一种高度抽象,表示了一个函数式编程思想。
swift 本身就支持函数式编程,而在 swift 语言中,() 可以表示为一个空的 元组,而元组可以表示任何的值。

  • 0-Tuple表示空值,也是Void的别名。
  • 1-Tuple表示任意类型的实例,如(Int,String,Enum,Struct等等),也就是 Int 可以看作是一个 (Int) ,String 是一个(String),以此类推。
  • N-Tuple表示多个值的组合,如 (2, "A")(2, "B", Struct)等。

Tuple的详细知识请参考 Tuples(Medium Link)

在编程中,数据运算 这两核心就可以在 swift 中抽象成 () -> ()了,一切的数据都可以被抽象成 (), 一切的运算过程都被抽象成 ->,那么之前认为->只不过是新语言标新立异于古老的 OC 而放在后面的想法就是不成立的了,-> 表示数据流动的方向,从一个数据到另一个数据。

到RAC中去

熟知RAC的同学都知道,RAC中有三种Event,分别为nexterrorcompleted。而RAC中的Signal(以RACSignal为代表)会发送事件流(a stream of events)订阅者(Subscribers)。不熟悉的请先了解RAC基本用法

同样以RAC基本用法这篇文章里的数据流动示例图作为参考,
整个Signal的传递过程可以抽象成(() -> ()) -> (() -> ()) -> ... -> (() -> ())

而RAC的作者Justin Spahr-Summers 在 GitHub Reactive Cocoa Developer Conference 上的《The Future Of ReactiveCocoa》(YTB视频)参考.NET的四种Interface

  • IEnumerable
  • IEnumerator
  • IObservable
  • IObserver

将函数式编程中的Event分为了:

这四种抽象的范式的表示方式和上文中通过 swift 抽象的 () -> () 有异曲同工之妙。

将上文中的进行一下操作:

1> (() -> ()) -> (() -> ()) -> ... -> (() -> ())
2> ‘1>’ 表达式中 () 使用 Event 替换
3> (Event1 -> ()) -> (Event2 -> ()) -> ... -> (EventN -> ())
4> ‘3>’表达式等同于 Observable 的抽象。

Observer: Event -> ()

Observer 等同于RAC中的 subscriber,当接收到一个Event时,就执行某些操作。类似于 swift 中的 Sink 协议。

Observable: (Event -> ()) -> ()

在函数式编程中,函数作为一等公民,可以作为值传递,所以Observable(RAC中的 Signal) 可以接受 Observer,当 Observer 在值传递过程中,也会执行某些操作。

Enumerator: () -> Event

到目前为止,这些都很好理解,然而硬币也有两面性,正如黑格尔的奥伏赫变一般,Observer 翻转 -> 的方向,得到 () -> Event
Enumerator,当其发生交互时,返回一个 Event。很类似于 swift 中的 Generator 协议。

Enumerable: () -> (() -> Event)

同样翻转Observable->得到 () -> (() -> Event),即Enumerable。一个Enumerable(类似于collection)可以创建一组Enumerator,十分接近于 swift 中的 Sequence 协议。

Push & Pull

  • ObserverObservable属于 Push 类型的,是生产者驱动的。
  • EnumeratorEnumerable 属于 Pull 类型的,是消费者驱动的。

Enumerator 升级

() -> Event 因为调用者必须得到一个Event,所以只能通过同步的方式去检索值,就造成阻塞当前线程,这是不好的方式。

() -> Promise Event

() -> Promise Event 实现值通过异步的方式获取,封装每个结果到Promise中,然后在未来的某个时间点给我们提供Event
同理,() -> (() -> Event) 变为 () -> (() -> Promise Event)
Promises & Futures

Promise:

  • Promise 确保任务能执行
  • 提供了无序和异步的功能
  • 由调用者完全控制

Promise 代表了一种 延时计算 的思想,与 swift 和 Golang 中的 defer 提供的功能一样。举个例子,消费者可以立刻获取5个 Promises,但是这样就做了无用功,因为只需要从第5个开始,跳过前4个,这样前4个promise就根本不会触发,消费者完全控制。

总结

Observables 和 Enumerables 是:

  • 一元运算(Monadic)
  • 模块化(Modular)
  • 异步的(Asynchronous)

Observables 对任意一个 observers 都是相同的。
Enumerables 对每一次枚举完全独立的。
Observables 永远都是活跃的(live)。
Enumerables 每一次的枚举都会开启新的工作(work)。

忘记Hot Signal 和 Cold Signal 吧!

Hot Signal Observables
Cold Signal Enumerables

本文根据 RAC 的作者在 2014 年的 GitHub Reactive Cocoa Developer Conference 上的《The Future Of ReactiveCocoa》(YTB视频)Talk 及 sunnyxx的博客文章:《() -> ()》 ,从 swift 的语法开始,对 RAC 的一些抽象进行了整理,试图去理解函数式编程的思想,如有出入,望不吝斧正(:/手动感谢)。

参考



对《The future of ReactiveCocoa》的一些思考的更多相关文章

  1. The Future Of ReactiveCocoa by Justin Spahr-Summers

    https://www.bilibili.com/video/av9783052?from=search&seid=14165903430339282774

  2. iOS开发 ReactiveCocoa入门教程 第二部分

    ReactiveCocoa 是一个框架,它允许你在你的iOS程序中使用函数响应式(FRP)技术.加上第一部分的讲解,你将会学会如何使用信号量(对事件发出数据流)如何替代标准的动作和事件处理逻辑.你也会 ...

  3. ReactiveCocoa入门教程:第二部分

    翻译自:http://www.raywenderlich.com/62796/reactivecocoa-tutorial-pt2 原文链接: ReactiveCocoa 是一个框架,它允许你在你的i ...

  4. ReactiveCocoa入门教程--第二部分

    翻译自:http://www.raywenderlich.com/62796/reactivecocoa-tutorial-pt2 ReactiveCocoa 是一个框架,它允许你在你的iOS程序中使 ...

  5. ReactiveCocoa 迎接下一个更加美好的世界

    什么是ReactiveCocoa 如果你有看Github的Trending Objective-C榜单,那你肯定是见过ReactiveCocoa了.如果你在weibo上关注唐巧.onevcat等国内一 ...

  6. 面向未来的友好设计:Future Friendly

    一年前翻译了本文的一部分,最近终于翻译完成.虽然此设计思想的提出已经好几年了,但是还是觉得应该在国内推广一下,让大家知道“内容策略”,“移动优先”,“响应式设计”,“原子设计”等设计思想和技术的根源. ...

  7. ReactiveCocoa代码实践之-更多思考

    三.ReactiveCocoa代码实践之-更多思考 1. RACObserve()宏形参写法的区别 之前写代码考虑过 RACObserve(self.timeLabel , text) 和 RACOb ...

  8. 博弈论揭示了深度学习的未来(译自:Game Theory Reveals the Future of Deep Learning)

    Game Theory Reveals the Future of Deep Learning Carlos E. Perez Deep Learning Patterns, Methodology ...

  9. 【iOS】小项目框架设计(ReactiveCocoa+MVVM+AFNetworking+FMDB)

    上一个项目使用到了ReactiveCocoa+MVVM+AFNetworking+FMDB框架设计,从最初的尝试,到后来不断思考和学习,现在对这样一个整体设计还是有了一定了理解与心得.在此与大家分享下 ...

随机推荐

  1. 关于Promise的异步依次函数调用

    在Promise中async用于定义一个异步函数(可不写),该函数返回一个Promise. 如果async函数返回的是一个同步的值,这个值将被包装成一个理解resolve的Promise, 等同于re ...

  2. python字符串常用函数

    # 索引与切片  *** capitalize()  **首字母大写 upper() lower() *** 大写和小写函数 startswith endswith  ***    判断以‘’字母’开 ...

  3. JAVAEE 和项目开发(第四课:HTTP的响应格式和响应状态码)

    HTTP 协议之响应 响应格式的结构: 响应行(状态行):HTTP 版本.状态码.状态消息 响应头:消息报头,客户端使用的附加信息 空行:响应头和响应实体之间的,必须的. 响应实体:正文,服务器返回给 ...

  4. Jenkins-在CentOS7 上安装配置

    Jenkins-在CentOS7 上安装配置 1.安装JDK yum install -y java 2.安装jenkins 方法1:添加Jenkins库到yum库,Jenkins将从这里下载安装. ...

  5. 自己整理的常用SQL Server 2005 语句、

    --创建数据库 create database 数据库 go --打开数据库 use 数据库 --删除数据库 drop database 数据库 Go --创建数据表 create table 数据表 ...

  6. 201903-2 二十四点 Java

    思路: 数据结构中,栈可以解决运算的问题.利用压栈和弹栈操作实现(这里用队列模拟).具体的: 遇到乘除号,弹出栈顶元素,将计算结果压入栈中.遇到加减号,将后面的数一起压入栈中. 注意: substri ...

  7. SpringBoot学习(四)——配置文件占位符

    RandomValuePropertySource:配置文件中可以使用随机数 ${Random.value}  ${random.int}, ${random.long}, ${random.int( ...

  8. c++ rand随机数生成(随机种子设置)

    需求:每次初始化不同的随机数 1.默认 //这样用每次都会产生相同数字 #include <stdlib.h> #include <stdio.h> #define N 10 ...

  9. Zookeeper--Zookeeper单机安装

    参考 https://www.cnblogs.com/lsdb/p/7297731.html https://zookeeper.apache.org/doc/r3.4.13/zookeeperSta ...

  10. CSS3新特性—animate动画

    1.animate介绍 1. @keyframes 自定义动画名称 { from { } to { } } 2. 通过动画名称调用动画集 animation-name: 动画集名称. 3. 属性介绍: ...