zoukankan      html  css  js  c++  java
  • [RxSwift教程]8、变换操作符:buffer、map、flatMap、scan等

    ★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★
    ➤微信公众号:山青咏芝(let_us_code)
    ➤博主域名:https://www.zengqiang.org
    ➤GitHub地址:https://github.com/strengthen/LeetCode
    ★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★★

    强烈推荐!!!试用博主个人App作品!提需求!提建议!
    App Store搜索:【Mind Draft
    中国区可直接点击跳转:【Mind Draft

    八、变换操作(Transforming Observables)

    变换操作指的是对原始的 Observable 序列进行一些转换,类似于 Swift 中 CollectionType 的各种转换。
     

    1,buffer

    (1)基本介绍
    • buffer 方法作用是缓冲组合,第一个参数是缓冲时间,第二个参数是缓冲个数,第三个参数是线程。
    • 该方法简单来说就是缓存 Observable 中发出的新元素,当元素达到某个数量,或者经过了特定的时间,它就会将这个元素集合发送出来。
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)
    (2)使用样例
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    import UIKit
    import RxSwift
    import RxCocoa
     
    class ViewControllerUIViewController {
         
        let disposeBag = DisposeBag()
         
        override func viewDidLoad() {
     
            let subject = PublishSubject<String>()
     
            //每缓存3个元素则组合起来一起发出。
            //如果1秒钟内不够3个也会发出(有几个发几个,一个都没有发空数组 [])
            subject
                .buffer(timeSpan: 1, count: 3, scheduler: MainScheduler.instance)
                .subscribe(onNext: { print($0) })
                .disposed(by: disposeBag)
     
            subject.onNext("a")
            subject.onNext("b")
            subject.onNext("c")
             
            subject.onNext("1")
            subject.onNext("2")
            subject.onNext("3")
        }
    }
    运行结果如下:
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)

    2,window

    (1)基本介绍
    • window 操作符和 buffer 十分相似。不过 buffer 是周期性的将缓存的元素集合发送出来,而 window 周期性的将元素集合以 Observable 的形态发送出来。
    • 同时 buffer 要等到元素搜集完毕后,才会发出元素序列。而 window 可以实时发出元素序列。
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)
    (2)使用样例
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    import UIKit
    import RxSwift
    import RxCocoa
     
    class ViewControllerUIViewController {
         
        let disposeBag = DisposeBag()
         
        override func viewDidLoad() {
             
            let subject = PublishSubject<String>()
             
            //每3个元素作为一个子Observable发出。
            subject
                .window(timeSpan: 1, count: 3, scheduler: MainScheduler.instance)
                .subscribe(onNext: { [weak self]  in
                    print("subscribe: ($0)")
                    $0.asObservable()
                        .subscribe(onNext: { print($0) })
                        .disposed(by: self!.disposeBag)
                })
                .disposed(by: disposeBag)
             
            subject.onNext("a")
            subject.onNext("b")
            subject.onNext("c")
             
            subject.onNext("1")
            subject.onNext("2")
            subject.onNext("3")
        }
    }

    运行结果如下: 

    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)

    3,map

    (1)基本介绍
    • 该操作符通过传入一个函数闭包把原来的 Observable 序列转变为一个新的 Observable 序列。
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)
    (2)使用样例
    1
    2
    3
    4
    5
    6
    let disposeBag = DisposeBag()
     
    Observable.of(1, 2, 3)
        .map { $0 * 10}
        .subscribe(onNext: { print($0) })
        .disposed(by: disposeBag)
    运行结果如下:
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)

    4,flatMap

    (1)基本介绍
    • map 在做转换的时候容易出现“升维”的情况。即转变之后,从一个序列变成了一个序列的序列。
    • 而 flatMap 操作符会对源 Observable 的每一个元素应用一个转换方法,将他们转换成 Observables。 然后将这些 Observables 的元素合并之后再发送出来。即又将其 "拍扁"(降维)成一个 Observable 序列。
    • 这个操作符是非常有用的。比如当 Observable 的元素本生拥有其他的 Observable 时,我们可以将所有子 Observables 的元素发送出来。
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)
    (2)使用样例
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    let disposeBag = DisposeBag()
     
    let subject1 = BehaviorSubject(value: "A")
    let subject2 = BehaviorSubject(value: "1")
     
    let variable = Variable(subject1)
     
    variable.asObservable()
        .flatMap { $0 }
        .subscribe(onNext: { print($0) })
        .disposed(by: disposeBag)
     
    subject1.onNext("B")
    variable.value = subject2
    subject2.onNext("2")
    subject1.onNext("C")
    运行结果如下:
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)

    5,flatMapLatest

    (1)基本介绍
    • flatMapLatest 与 flatMap 的唯一区别是:flatMapLatest 只会接收最新的 value 事件。
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)
     
    (2)这里我们将上例中的 flatMap 改为 flatMapLatest。
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    let disposeBag = DisposeBag()
     
    let subject1 = BehaviorSubject(value: "A")
    let subject2 = BehaviorSubject(value: "1")
     
    let variable = Variable(subject1)
     
    variable.asObservable()
        .flatMapLatest { $0 }
        .subscribe(onNext: { print($0) })
        .disposed(by: disposeBag)
     
    subject1.onNext("B")
    variable.value = subject2
    subject2.onNext("2")
    subject1.onNext("C")
    运行结果如下:
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)
     

    6,flatMapFirst

    (1)基本介绍
    • flatMapFirst 与 flatMapLatest 正好相反:flatMapFirst 只会接收最初的 value 事件。
    该操作符可以防止重复请求:
    比如点击一个按钮发送一个请求,当该请求完成前,该按钮点击都不应该继续发送请求。便可该使用 flatMapFirst 操作符。
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)

    (2)这里我们将上例中的 flatMapLatest 改为 flatMapFirst。
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    let disposeBag = DisposeBag()
     
    let subject1 = BehaviorSubject(value: "A")
    let subject2 = BehaviorSubject(value: "1")
     
    let variable = Variable(subject1)
     
    variable.asObservable()
        .flatMapFirst { $0 }
        .subscribe(onNext: { print($0) })
        .disposed(by: disposeBag)
     
    subject1.onNext("B")
    variable.value = subject2
    subject2.onNext("2")
    subject1.onNext("C")
    运行结果如下: 
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)

    7,concatMap

    (1)基本介绍
    • concatMap 与 flatMap 的唯一区别是:当前一个 Observable 元素发送完毕后,后一个Observable 才可以开始发出元素。或者说等待前一个 Observable 产生完成事件后,才对后一个 Observable 进行订阅。
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)
    (2)这里我们将“样例3”中的 flatMap 改为 concatMap。
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    let disposeBag = DisposeBag()
     
    let subject1 = BehaviorSubject(value: "A")
    let subject2 = BehaviorSubject(value: "1")
     
    let variable = Variable(subject1)
     
    variable.asObservable()
        .concatMap { $0 }
        .subscribe(onNext: { print($0) })
        .disposed(by: disposeBag)
     
    subject1.onNext("B")
    variable.value = subject2
    subject2.onNext("2")
    subject1.onNext("C")
    subject1.onCompleted() //只有前一个序列结束后,才能接收下一个序列
    运行结果如下:
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)

    8,scan

    (1)基本介绍
    • scan 就是先给一个初始化的数,然后不断的拿前一个结果和最新的值进行处理操作。
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)
    (2)使用样例
    1
    2
    3
    4
    5
    6
    7
    8
    let disposeBag = DisposeBag()
     
    Observable.of(1, 2, 3, 4, 5)
        .scan(0) { acum, elem in
            acum + elem
        }
        .subscribe(onNext: { print($0) })
        .disposed(by: disposeBag)
    运行结果如下:
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)
     

    9,groupBy

    (1)基本介绍
    • groupBy 操作符将源 Observable 分解为多个子 Observable,然后将这些子 Observable 发送出来。
    • 也就是说该操作符会将元素通过某个键进行分组,然后将分组后的元素序列以 Observable 的形态发送出来。
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)
    (2)使用样例
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    let disposeBag = DisposeBag()
     
    //将奇数偶数分成两组
    Observable<Int>.of(0, 1, 2, 3, 4, 5)
        .groupBy(keySelector: { (element) -> String in
            return element % 2 == 0 ? "偶数" "基数"
        })
        .subscribe { (event) in
            switch event {
            case .next(let group):
                group.asObservable().subscribe({ (event) in
                    print("key:(group.key)    event:(event)")
                })
                .disposed(by: disposeBag)
            default:
                print("")
            }
        }
    .disposed(by: disposeBag)
    运行结果如下:
    原文:Swift - RxSwift的使用详解8(变换操作符:buffer、map、flatMap、scan等)
  • 相关阅读:
    临时表的问题
    List集合和Set集合互转
    mysql数据库事件
    mysql存储过程事务
    N皇后问题
    递归实现字符数组的全排列及组合
    判断一个序列是否为某二叉搜索树的后续遍历结果
    递归实现两个有序链表的合并
    递归实现字符串反转
    根据字节数截取字符串
  • 原文地址:https://www.cnblogs.com/strengthen/p/13673473.html
Copyright © 2011-2022 走看看