RxSwift-初探

函数式编程简称 FP(Functional Programming) ,函数式编程就是一种抽象程度很高的编程范式,它将计算机运算看做是数学中函数的计算,而纯粹的函数式编程语言编写的函数没有变量,因此,任意一个函数,只要输入是确定的,输出就是确定的,这种纯函数我们称之为没有副作用。而允许使用变量的程序设计语言,由于函数内部的变量状态不确定,同样的输入,可能得到不同的输出,因此,这种函数是有副作用的。

函数式编程的一个特点就是:允许把函数本身作为参数传入另一个函数,同时还允许返回一个函数!

函数表达式: y = f(x) —> x = f(x) —> y = f(f(x))

常规需求例子,感受函数式的舒服!

  • 对于数组 [1,2,3,4,5,6,7,8,9,10] ,首先获取 > 2 的数字,

  • 获取到的数字之后 + 1,

  • 再输出所有数字中的偶数

let array = [1,2,3,4,5,6,7,8,9,10]for num in array{if num > 2{let number = num + 1if (number % 2 == 0) {print(number)
                  }
            }
        }复制代码
  • 上面的代码嵌套层次之深,我觉得已经非常恶心

  • 代码的可读性也是非常差的

  • 代码复用性较低

  • 维护以及代码构建成本太高,这样就成了函数式出现的必要性

let array = [1,2,3,4,5,6,7,8,9,10]
array.filter{ $0 > 2}
            .filter{ ($0+1) % 2 == 0 }
            .forEach { print($0) }复制代码
  • 代码层次非常清晰

  • 代码可读性高

  • 代码复用性高

  • 代码简洁,直接面向开发需求

二:响应式

对象对某一数据流变化做出响应的这种编码方式称为响应式。例如我们在爱奇艺平台观看视频,我们只需要在某一个时刻订阅了这个视频,后面平台自媒体运营者不断更新视频,我们随着时间也能自定接受推送,这就是响应

在iOS开发中我们经常会响应一些事件 button、tap、textField、textView、notifaction、KVO、NSTimer 等等这些,都需要做响应监听,响应后都需要在对应的响应事件中去做处理,而原生开发中,触发对象与响应方法是分离的,如 button 的初始化和点击响应方法是分离的。功能代码与逻辑代码分析,导致很多比较恶心的感觉!

blingbling, blingbling, blingbling! RxSwift 闪亮登场

RxSwift简介

RxSwiftReactiveX 家族的重要一员, ReactiveXReactive Extensions 的缩写,一般简写为 RxReactiveX 官方给 Rx 的定义是: Rx是一个使用可观察数据流进行异步编程的编程接口。

ReactiveX 不仅仅是一个编程接口,它是一种编程思想的突破,它影响了许多其它的程序库和框架以及编程语言。它拓展了 观察者模式 ,使你能够 自由组合多个异步事件 ,而 不需要去关心线程同步,线程安全并发数据以及I/O阻塞

RxSwiftRxSwift 语言开发的一门函数响应式编程语言, 它可以代替iOS系统的 Target Action / 代理 / 闭包 / 通知 / KVO ,同时还提供 网络数据绑定UI事件处理UI的展示和更新多线程 ……

鉴于 Swift 日渐增长的影响力,ios开发者不可避免的要学习和使用 Swift 这门语言进行编程开发。而 RxSwift 对使用 Swift 的帮助有如下几点:

Swift 为值类型,在传值与方法回调上有影响, RxSwift 一定程度上弥补 Swift 的灵活性

  • RxSwift 使得代码复用性较强,减少代码量

  • RxSwift 因为声明都是不可变更,增加代码可读性

  • RxSwift 使得更易于理解业务代码,抽象异步编程,统一代码风格

  • RxSwift 使得代码更易于编写集成单元测试,增加代码稳定性

一:button点击事件

//MARK: - RxSwift应用-button响应
func setupButton() {
    // 传统UI事件
    self.button.addTarget(self, action: #selector(didClickButton), for: .touchUpInside)
    // 这样的操作 - 不行啊!代码逻辑与事件逻辑分层
    self.button.rx.tap
        .subscribe(onNext: { [weak self] inprint("点了,小鸡炖蘑菇")
            self?.view.backgroundColor = UIColor.orange   
        })
        .disposed(by: disposeBag) 
}复制代码

二:textfiled文本响应

//MARK: - RxSwift应用-textfiled
func setupTextFiled() {
    // 我们如果要对输入的文本进行操作 - 比如输入的的内容 然后我们获取里面的偶数
    // self.textFiled.delegate = self
    // 感觉是不是特别恶心
    // 下面我们来看看Rx
    self.textFiled.rx.text.orEmpty.changed.subscribe(onNext: { (text) inprint("监听到了 - \(text)")
    }).disposed(by: disposeBag)
    
    self.textFiled.rx.text.bind(to: self.button.rx.title()).disposed(by: disposeBag)
}复制代码

三:scrollView效果

//MARK: - RxSwift应用-scrollView
func setupScrollerView() {
    scrollView.rx.contentOffset.subscribe(onNext: { [weak self] (content) inself?.view.backgroundColor = UIColor.init(red: content.y/255.0*0.8, green: content.y/255.0*0.3, blue: content.y/255.0*0.6, alpha: 1);print(content.y)
    }).disposed(by: disposeBag)
}复制代码
  • 从上面三个UI效果来说,如果原生实现必将做很多恶心的处理: addTarget 或者 实现响应代理 , 但是在 RxSwift 的世界里只需要一句代码: 爽!

四:KVO

 //MARK: - RxSwift应用-KVO
func setupKVO() {
    // 系统KVO 还是比较麻烦的
    // person.addObserver(self, forKeyPath: "name", options: .new, context: nil)
    person.rx.observeWeakly(String.self, "name").subscribe(onNext: { (change) inprint(change ?? "helloword")
    }).disposed(by: disposeBag)
}复制代码
  • 原生KVO就需要三部曲,非常恶心,其实对于我们开发来说根本没必要!

  • RxSwift 还是温柔的一句代码搞定

五:通知

//MARK: - 通知
func setupNotification(){
    NotificationCenter.default.rx
        .notification(UIResponder.keyboardWillShowNotification)
        .subscribe { (event) inprint(event)
    }.disposed(by: disposeBag)
}复制代码
  • 通知的实现,也是比较友善的:编程习惯都有在 RxSwift 风格里面保留

  • 这样能够最大的降低开发者的门槛,也是一个优秀框架的必备条件

六:手势

//MARK: - 手势
func setupGestureRecognizer(){let tap = UITapGestureRecognizer()
    self.label.addGestureRecognizer(tap)
    self.label.isUserInteractionEnabled = truetap.rx.event.subscribe { (event) inprint("点了label")
    }.disposed(by: disposeBag)  
}复制代码
  • 很多小伙伴可能会吐槽手势怎么不那么爽!其实你误解, RxSwift 是实现的响应关系,本身手势也是需要添加事件对象的!

  • 当然这里我个人建议:可以更多拓展一层!降低代码量

  • 感觉 RxSwift 会把开发者变傻,哈哈哈……

七:网络请求

//MARK: - RxSwift应用-网络请求
func setupNextwork() {let url = URL(string: "https://www.baidu.com")
    URLSession.shared.rx.response(request: URLRequest(url: url!))
        .subscribe(onNext: { (response, data) inprint("response ==== \(response)")print("data ===== \(data)")
        }, onError: { (error) inprint("error ===== \(error)")
        }).disposed(by: disposeBag)
}复制代码
  • 针对 URLSession 也是封装处理。提供了 response , data , json 不同的便于开发者选择的合适函数,说白就是方便

  • 但是针对网络这一层,我们一般开发肯定是要处理,不可能这么简单(毕竟不符合开发需求)

  • 其实 RxSwift 在网络这一层也是有很多响应式处理:比如典型的 RxMoya

八:timer定时器

//MARK: - RxSwift应用-timer定时器
func setupTimer() {
    timer = Observable.interval(1, scheduler: MainScheduler.instance)
    timer.subscribe(onNext: { (num) inprint("hello word \(num)")
    }).disposed(by: disposeBag)
}复制代码
  • RxSwift 实现的 timer 免去了我们计时器的一些不必要的麻烦

  • runloop影响

  • 销毁问题

  • 线程问题

RxSwift这个框架非常牛逼,把函数响应式玩到了高潮!无论从代码量,还是从代码的可读性,抑或代码的复用性……都是大大优化!让开发者抛弃更多的胶水代码,直面开发需求。如果你还在为你开发效率而烦恼,那么 RxSwift 是你不二的选择!如果还在纠结怎么学习,那么我的这个 RxSwift 专题就是你不二的选择!如果你想更加系统学习进阶iOS,那么Cooci更是你不二的选择!

就问此时此刻还有谁?45度仰望天空,该死!我这无处安放的魅力!