Hello ReactiveSwift(2): 示例:在线搜索 ——(简译)

官方文档:
http://reactivecocoa.io/reactiveswift/docs/latest/index.html
实战项目:
https://github.com/JornWu/ZhiBo_Swift.git


<i>需求:假设你有一个TextField,并且每当用户输入内容时,你都希望创建一个搜索该查询的网络请求。</i>

1、观察文本编辑

第一步是监视该textfield的编辑,用到UITextField的专门用于该目的的RAC拓展:

let searchStrings = textField.reactive.continuousTextValues

这给我们一个发送值类型为String?的Signal

2、制作网络请求

对于每个字符串,我们要执行一个网络请求。ReactiveSwift提供了一个 URLSession扩展功能:

let searchResults = searchStrings
    .flatMap(.latest) { (query: String?) -> SignalProducer<(Data, URLResponse), AnyError> in
        let request = self.makeSearchRequest(escapedQuery: query)
        return URLSession.shared.reactive.data(with: request)
    }
    .map { (data, response) -> [SearchResult] in
        let string = String(data: data, encoding: .utf8)!
        return self.searchResults(fromJSONString: string)
    }
    .observe(on: UIScheduler())

这将String的制造器转变成包含搜索结果的Array制造器。且在主线程上进行(使用UISchedule)。此外,
flatMap(.latest)
这里确保只有一个搜索(最新的) 被允许运行。如果用户在网络请求仍处于进行状态时键入另一个字符,则在启动新的搜索之前将被取消。试想想自己手动去实现将花费到少代码。

3、接受结果

由于搜索字符串的来源是具有热信号语义的Signal,我们应用的变换会自动进行推测每当任何时候新值从searchStrings发出时。
因此,我们可以使用Signal.observe(_:)简单地观察Signal

searchResults.observe { event in
    switch event {
    case let .value(results):
        print("Search results: \(results)")

    case let .failed(error):
        print("Search error: \(error)")

    case .completed, .interrupted:
        break
    }
}

这里,只要打印到控制台,我们可以观察到包含我们的结果的Value,也可以很容易地通过其他操作。如在屏幕上更新tableView或label。

4、处理错误

目前为止在这个示例中,任何网络错误都将会产生一个Failed事件,这些事件将会终止事件流,不幸的是,这意味着以后的请求将不会被执行。
为了解决这个问题,我们需要决定如何处理发生的故障。最快的解决方案是记录它们,然后忽略它们:

.flatMap(.latest) { (query: String) -> SignalProducer<(Data, URLResponse), AnyError> in
        let request = self.makeSearchRequest(escapedQuery: query)

        return URLSession.shared.reactive
            .data(with: request)
            .flatMapError { error in
                print("Network error occurred: \(error)")
                return SignalProducer.empty
            }
    }

通过使用empty事件流来替代failures,我们可以有效地忽略它们。但在丢弃前,可能要做至少两次适当的尝试。简单的,有有个尝试操作恰好可以这样做。
我们优化的searchResults制造者可以看起来如下:

let searchResults = searchStrings
    .flatMap(.latest) { (query: String) -> SignalProducer<(Data, URLResponse), AnyError> in
        let request = self.makeSearchRequest(escapedQuery: query)

        return URLSession.shared.reactive
            .data(with: request)
            .retry(upTo: 2)
            .flatMapError { error in
                print("Network error occurred: \(error)")
                return SignalProducer.empty
            }
    }
    .map { (data, response) -> [SearchResult] in
        let string = String(data: data, encoding: .utf8)!
        return self.searchResults(fromJSONString: string)
    }
    .observe(on: UIScheduler())

5、节流请求

现在,假如你只想偶尔地真正执行搜索,一最小化流量。
ReactiveCocoa有一个声明的throttle运算符,我们可以应用于我们的搜索字符串:

let searchStrings = textField.reactive.continuousTextValues
    .throttle(0.5, on: QueueScheduler.main)

这可以阻止发送间隔不到0.5秒的值。
手动地这样实现将要求具有有效的状体,且最终代码会更难理解!使用ReactiveCocoa,我们可以只使用一个运算符将时间整合到我们的事件流中即可。

6、调试事件流

由于其本质,一个流的堆链可能有几十个帧,这通常使调试成为非常令人沮丧的活动,一种天真的调试方式,就是将流行注入副作用,就像这样:

let searchString = textField.reactive.continuousTextValues
    .throttle(0.5, on: QueueScheduler.main)
    .on(event: { print ($0) }) // the side effect

这将打印流的事件,同时保留原始流的行为。SignalProducerSignal都提供logEvents操作,这将自动为您做到这一点:

let searchString = textField.reactive.continuousTextValues
    .throttle(0.5, on: QueueScheduler.main)
    .logEvents()

有关更多信息和预先的用法,请查看
Debugging Techniques 文档。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,088评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,715评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,361评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,099评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 60,987评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,063评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,486评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,175评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,440评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,518评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,305评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,190评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,550评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,880评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,152评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,451评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,637评论 2 335

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,490评论 18 139
  • RAC使用测试Demo下载:github.com/FuWees/WPRACTestDemo 1.ReactiveC...
    FuWees阅读 6,338评论 3 10
  • 看《钢琴师》的时候,印象最深刻的一个片段是,钢琴家Szpilman在四处躲藏纳粹时,曾住进曾是相爱的心上人、如今已...
    淑子是永动机阅读 255评论 0 0
  • 一、前置索引的好处 使用前置索引的好处 Normally, the entire value of a colum...
    AQ王浩阅读 3,817评论 0 6