iOS的函数响应型编程

流和序列

流是值的序列化的抽象,你可以认为一个流就像一条水管,而值就是流淌在水管中的水,值从管道的一端流入从另一端流出。当值从管道的另一端流出的时候,我们可以读取过去所有的值,甚至是刚刚进入管道的值(即当前值)。接下来让我们拭目以待! 呃,值的序列化,那是什么鬼?以我们当前的认知水平来说,她就像是一个数组,一个列表。事实上,使用rac_sequeuece我们能够轻松地将数组转化为一个流:

NSArray *array = @[ @1, @2, @3 ];
RACSequence * stream = [array rac_sequence];

等一下!Sequences?我以为我们在处理Stream? 好吧,说明一下,Sequences是两种特定类型的流的一种,实际上,RACSequence是一个RACStream的子类。 我们能用流做什么呢?好吧,我将使用流来展示上一章中提到的例子。应用在平方数映射上:

[stream map:^id (id value){
    return @(pow([value integerValue], 2));
}];

注意,跟数组一样,流不能包含nil元素。[译者注:NSArray中以nil作为结束标示,stream也一样]。 非常好!但是流映射后还是流,我们怎么样才能得到数组呢?幸运的是,RACSequence有一个方法返回数组:array

NSLog(@"%@",[stream array]);

这会打印映射后的数组。比起直接使用RXCollections这多出了几个步骤,但这里我只想说明使用流也可以达成任务。

当然,我们可以合并上面的方法调用来避免污染变量的作用域.

NSLog(@"%@",[[[array rac_sequence] map:^id (id value){
                    return @(pow([value integerValue], 2));
                }] array]);

总的来说,我们做了这样的事情:

  • 将数组转化成一个序列类型的流。
  • 对流进行映射得到一个新的流。
  • 将新的流转为数组。

序列,默认情况下是延迟加载的(也称:懒加载或被动加载),是pull-driven的,在他们被生成的时候就会提供确切的值,而数组方法会强制给序列的每一个成员赋值。

我们来看一下filtering。为了使用ReactiveCocoa来过滤我们的数组,我们需要再一次把它序列化以便于使用过滤。

NSLog(@"%@", [[[array rac_sequence] filter:^BOOL (id value){
                        return [value integerValue] % 2 == 0;
                    }] array]);

最后看一下怎么让一个序列流合并为单个值(folding):

NSLog(@"%@",[[[array rac_sequence] map:^id (id value){
                    return [value stringValue];
                }] foldLeftWithStart:@"" reduce:^id (id accumulator, id value){
                    return [accumulator stringByAppendingString:value];
            }]);

这种情况下,我们在序列上进行了链式调用,当我们讨论下一节'信号'的时候,(链式调用)是一个关键的概念。

ReactiveCocoa具有左折叠和右折叠的概念。左折叠时折叠算法将从头到尾遍历数组,反之称为右折叠。这样的命名(即左、右折叠)暗示了编程语言对列表的理解,这种概念在Objective-C中是没有的。

确定你现在已经理解了到此为止我们所说的内容,这对后面将要进行的讲解非常重要。