RxSwift之路 1#Swift语法知识准备

在开始学习 RxSwift 之前,一定要对 Swift 相关语法有所了解,否则就很难理解为什么可以这样。关于 Swift 的学习其实只要看看 Swift 的官方文档就可够了。我之前也列过一些学习资源:来自一线开发者的Swift学习资源推荐
现在开始进入正题。

Swift的优势

想一个有趣的问题,为什么没有 RxObjc 呢?
实际上响应式的编程框架对语言还是有些要求的,当然 OC 确实也有一个奠基式的 FRP 框架 ReactiveCocoa。但是客观的说,在 Swift 里响应式的框架写起来会愉快的多,或者说更能发挥出语言的优势。
Swift 契合响应式有以下几点:

  • 强类型,强类型的泛型
  • 灵活的闭包
  • 对并发模型的原生支持(Swift 5 的特性,还未发布

Enum的关联值和泛型

Swift 中的枚举(Enum)能力相比 OC 可以说得到了升华。不再只是一个类似预编译时的宏,而是一个完整的类型。和 Struct 一样,可以给他定义初始化方法,声明函数,添加扩展。同样的泛型同样也试用于 Enum。
枚举还有一项神奇的能力叫关联值。一个枚举可以的值可以是不同的类型。官方手册里的示例代码如下:

enum ServerResponse {
case result(String, String)
case failure(String)
} let success = ServerResponse.result("6:00 am", "8:09 pm")
let failure = ServerResponse.failure("Out of cheese.") switch success {
case let .result(sunrise, sunset):
print("Sunrise is at \(sunrise) and sunset is at \(sunset).")
case let .failure(message):
print("Failure... \(message)")
}

每个 case 可以携带不同的类型,并且可以不止是一个值。
当 Enum 结合泛型,就发生了有趣的事。直接贴代码:

enum OptionalValue<Wrapped> {
case none
case some(Wrapped)
} var possibleInteger: OptionalValue<Int> = .none
possibleInteger = .some()
这就是 Swift 中的 Optional 实现的类似代码。使用枚举实现,表示值有两种可能:没有值的 .none 和是 Wrapped 类型的 .some。

有了以上的知识我们再来看 Rx 中的事件流中的值Event

public enum Event<Element> {
/// Next element is produced.
case next(Element) /// Sequence terminated with an error.
case error(Swift.Error) /// Sequence completed successfully.
case completed
}

每一次事件的值有三种可能:1.值(next),2.完成结束(completed),3.失败(error)。

函数的默认参数

如果函数在声明时设置了一个默认值,那么在调用时这个参数就可以不传。
假设我们给 Int 定义个扩展方法increment。如果不传入值则默认加1,如果传入就按照传入的值增加:

extension Int {
func increment(with number: Int = ) -> Int {
return self + number
}
}
使用的时候 Xcode 就会提示两个函数:

不再需要像 OC 一样定义几个函数。
Rx 表示订阅的subscribe函数,有时只要写onNext,有时只要写onError,就是因为这个函数在声明时同时指定了默认参数:

extension ObservableType {

   public func subscribe(file: String = #file,line: UInt = #line,
function: String = #function,
onNext: ((E) -> Void)? = nil,
onError: ((Swift.Error) -> Void)? = nil,
onCompleted: (() -> Void)? = nil,
onDisposed: (() -> Void)? = nil)-> Disposable {
// ...
}

可以看到这个函数为订阅每个事件可能都声明了默认参数,所以你在订阅时可以只订阅自己关注的订阅。

闭包

闭包的使用类似 OC 中的 block,具体使用就不再介绍。提一下新手很容易忽略的语法糖。

闭包简化语法

  • 闭包参数、返回值类型自动推断
    因为 Swift 是强类型语言,闭包的参数的类型可以推断出来。比如如下代码:

    reversedNames = names.sorted(by: { (s1: String, s2: String) -> Bool in
    return s1 > s2
    })
    s1 和 s2 的类型和返回值类型 Xcode 可以推断出来可以省略:
    reversedNames = names.sorted(by: { (s1, s2) in
    return s1 > s2
    })
  • 闭包表达式只有一行,可以省略return
    因为只有一行,表达式执行的结果肯定就是要返回的值,所以可以省略。接上面的代码可以优化成这样:
    reversedNames = names.sorted(by: {
    (s1, s2) in s1 > s2
    })
  • 用$N代替参数
    有时实参的参数名没什么用,可以直接用$N代表实参,$0代表第一个参数。上面的代码进行这样的简化后就写成了这样:
    reversedNames = names.sorted(by:  { $ > $ })
  • 二元运算时可以直接省略只用操作符表示
    当表达式只有一行时,二元运算的表达式可以直接省略到只剩一个运算符,上面的可以省略为:
    reversedNames = names.sorted(by:  >)

尾闭包

当参数的最好一个参数是闭包时,可以直接把最后一个闭包的实现跟在函数后面。
直接贴代码表示:

// 一个最后一个参数是闭包的函数
func someFunctionThatTakesAClosure(closure: () -> Void) {
// function body goes here
} // 默认的调用方法
someFunctionThatTakesAClosure(closure: {
// closure's body goes here
}) // 省略最后一个参数的方法名,并且把闭包的实现直接跟在函数后面
someFunctionThatTakesAClosure() {
// trailing closure's body goes here
}
这种写法简化了代码,让代码看起来更简洁。 Rx 中的数据流操作符可以灵活的组织闭包,经常会用到简化的闭包的语法。 Observable.of(, , , , , )
.filter { $ % == }
.subscribe(onNext: {
print($)
})

刚开始的时候可能有点看不明白,慢慢的还原闭包的语法,之后看多了就会熟悉的。

RxSwift之路 1#Swift语法知识准备的更多相关文章

  1. RxSwift之路 2#如何开始

    RxSwift之路 2#如何开始 第一步当然是把项目clone到本地,github地址:https://github.com/ReactiveX/RxSwift. 官方文档 学习的第一手资源当然是项目 ...

  2. 那点你不知道的XHtml(Xml+Html)语法知识(DTD、XSD)

    什么是XHtml: 摘录网上的一句话,XHTML就是一个扮演着类似HTML的角色的XML. XHtml可当模板引擎应用: CYQ.Data 框架里有一套XHtmlAction模板引擎, 应用在QBlo ...

  3. Swift语法入门

    正文参考: https://developer.apple.com/library/prerelease/ios/documentation/Swift/Conceptual/Swift_Progra ...

  4. ios -- 教你如何轻松学习Swift语法(三) 完结篇

    前言:swift语法基础篇(二)来了,想学习swift的朋友可以拿去参考哦,有兴趣可以相互探讨,共同学习哦.      一.自动引用计数   1.自动引用计数工作机制      1.1 swift和o ...

  5. ios -- 教你如何轻松学习Swift语法(二)

    前言:swift语法基础篇(二)来了,想学习swift的朋友可以拿去参考哦,有兴趣可以相互探讨,共同学习哦.      一.可选类型(重点内容)   1.什么是可选类型?        1.1在OC开 ...

  6. ios -- 教你如何轻松学习Swift语法(一)

    目前随着公司开发模式的变更,swift也显得越发重要,相对来说,swift语言更加简洁,严谨.但对于我来说,感觉swift细节的处理很繁琐,可能是还没适应的缘故吧.基本每写一句代码,都要对变量的数据类 ...

  7. Swift翻译之-Swift语法入门 Swift语法介绍

    目录[-] Hello world - Swift 简单赋值 控制流 函数与闭包 对象和类 枚举与结构 协议和扩展 泛型 2014.6.3日,苹果公布最新编程语言Swift,Swift是一种新的编程语 ...

  8. Swift语法总结补充(一)

    Swift基础语法学习总结Swift高级语法学习总结Swift语法总结补充(一) 1. 可选类型是一种类型,String?就是Optional<String>,所以函数参数也可以声明为它2 ...

  9. Swift语法

    Swift语法 标签(空格分隔): Swift 1.打印输出语句 println("Hello, Swift!") 注意每行代码后面无需添加分号作为结束 2.简单值 let---常 ...

随机推荐

  1. C# 动态调用WebService 2

    using Microsoft.CSharp; using System; using System.CodeDom; using System.CodeDom.Compiler; using Sys ...

  2. 一个用SAM维护多个串的根号特技

    一个用SAM维护多个串的根号特技 基本介绍 在多个串的字符串题中,往往会出现一类题需要用到某个子串是否在一些母串中出现.此时对于 \(\text{parent}\) 树的 \(\text{right} ...

  3. 51Nod.1766.树上最远点对(树的直径 RMQ 线段树/ST表)

    题目链接 \(Description\) 给定一棵树.每次询问给定\(a\sim b,c\sim d\)两个下标区间,从这两个区间中各取一个点,使得这两个点距离最远.输出最远距离. \(n,q\leq ...

  4. 2602 ACM 杭电 骨头容器 01背包

    题意:装骨头的容器大小固定,有一堆骨头,已知骨头的价值和大小,在不超过容积大小的情况下,问:所装骨头的最大价值? 思路:典型的01背包问题,不需要有任何的变动. 模板: for(int j=v;j&g ...

  5. ACM/IOI 历年国家集训队论文集和论文算法分类整理

    国家集训队1999论文集 陈宏:<数据结构的选择与算法效率--从IOI98试题PICTURE谈起> 来煜坤:<把握本质,灵活运用--动态规划的深入探讨> 齐鑫:<搜索方法 ...

  6. Android AsyncTask将讲解

    原型:AsyncTask<Params, Progress, Result> Params 表示传入参数类型 Progress表示处理参数类型 Result表示返回类型 new Async ...

  7. C++程序设计方法3:default修饰符

    编译器自动生成的成员函数 如果以下成员函数用户都没有为类实现,则编译器会自动为类生成他们的缺省的实现 默认构造函数,空函数,什么也不做 析构函数,空函数,什么也不做: 拷贝构造函数-按bit位复制对象 ...

  8. bzoj 1005

    prufer序列 性质: 1.一棵n个结点的树可表示为长度 n-2 的prufer序列 2.每个结点出现在prufer序列中的次数==该结点的度 -1 公式推出来了,大数模板没有除法..等开学了Jav ...

  9. Python基础-修改excel、redis、接口开发、组织代码

    pymysql模块补充内容 1. 游标.description():显示表的字段属性 (什么是游标:游标用于交互式应用,就好比word里的光标一样,要修改某个地方,要先把光标移动到这里) 用好这个方法 ...

  10. JAVA自学笔记03

    1.三目运算符 1)格式:(关系表达式)?表达式1:表达式2 true则执行表达式1,false则执行表达式2 @ 例题1 :求两数中的较大值 System.out.println(x>y?x: ...