Swift常用基础知识(二)

Swift常用基础知识(一)

函数式编程

函数合成

func add1(_ v1: Int, _ v2: Int) -> Int { v1 + v2 }

func currying<A, B, C>(_ fn: @escaping (A, B) -> C) -> (B) -> (A) -> C {
    return { b in
        return { a in
            return fn(a, b)}
    }
}
print(currying(add1)(20)(10))


func add(_ v1: Int, _ v2: Int) -> Int { v1 + v2 }
func sub(_ v1: Int, _ v2: Int) -> Int { v1 - v2 }
func multiple(_ v1: Int, _ v2: Int) -> Int { v1 * v2 }
func divide(_ v1: Int, _ v2: Int) -> Int { v1 / v2 }
func mod(_ v1: Int, _ v2: Int) -> Int { v1 % v2 }

print(currying(add)(20)(100),
currying(sub)(20)(100),
currying(multiple)(20)(100),
currying(divide)(20)(100),
currying(mod)(20)(100))

柯里化(Currying)

prefix func ~<A, B, C>(_ fn: @escaping (A, B) -> C) -> (B) -> (A) -> C {
    { b in { a in return fn(a, b) } }
}

infix operator >>> : AdditionPrecedence
func >>><A, B, C>(_ fn1: @escaping (A) -> B, _ fn2: @escaping (B) -> C) -> (A) -> C {
    { fn2(fn1($0)) }
}
var num = 1
var fn = (~add)(3) >>> (~multiple)(5) >>> (~sub)(1) >>> (~mod)(10) >>> (~divide)(2)
print(fn(num))
print((((num+3)*5 - 1) % 10) / 2)

func add2(_ v1: Int, _ v2: Int, _ v3: Int) -> Int { v1 - v2 + v3 }
prefix func ~<A, B, C, D>(_ fn: @escaping (A, B, C) -> D) -> (C) -> (B) -> (A) -> D {
    { c in { b in { a in return fn(a, b, c) } } }
}
//20
print((~add2)(30)(20)(10)

函子(Functor)

  • 像Array、 Optional这样支持map允许的类型,称为函子(Functor)
public func map<T>(_ transform: (Element) throws -> T) rethrows -> [T]
public func map<U>(_ transform: (Wrapped) throws -> U) rethrows -> U?
  • Functor is the Type Justlike have this method
func map<T>(_ fn: (Inner) -> T) -> Type<T>

适用函子(Applicative Functor)

  • 对人员一个函子F,如果能支持一下运算,该函子就是一个适用函子
//func pure<T>(_ value: T) -> F<T>
//func <*><A, B>(_ fn: F<(A) -> B>, _ value: F<A>) -> F<B>
  • Array可以成为适用函子
func pure<T>(_ value: T) -> T? { value}
infix operator <*> : AdditionPrecedence
func <*><A, B>(_ fn: ((A) -> B)?, _ value: A?) -> B? {
    guard let f = fn, let v = value else { return nil}
    return f(v)
}

var value: Int? = 10
var fn: ((Int) -> Int)? = { $0 * 2 }
Optional(20)
print(fn <*> value as Any)

//Optional可以成为适用函子

func pure<T>(_ value: T) -> [T] { [value] }
func <*><A, B>(_ fn: [(A) -> B], _ value: [A]) -> [B] {
    var array: [B] = []
    if fn.count == value.count {
        for i in fn.startIndex..<fn.endIndex {
            array.append(fn[i](value[i]))
        }
    }
    return array
}
//[10]
print(pure(10))
var array = [{ $0 * 2}, { $0 + 10 }, { $0 - 5}] <*> [1, 2, 3]
//[2, 12, -2]
print(array)

单子(Monad)

  • 对任意一个类型F,如果嫩支持以下运算,那么就可以称为是一个单子(Monad)
func pure<T>(_ value: T) -> F<T>
func flatMap<A, B>(_ value: F<A>, _ fn: (A) -> F<B>) -> F<B>
  • 很显然Array、 Optional都是单子

面向协议编程

■ 面向协议编程( Protocol Oriented Programming ,简称POP )

  • 是Swift的一种编程范式,Apple于2015年WWDC提出
  • 在Swift的标准库中,能见到大量POP的影子
    ■ 同时, Swift也是一-门面向对象的编程语言( Object Oriented Programming ,简称OOP )
  • 在Swift开发中, OOP和POP是相辅相成的,任何一方并不能取代另一方
    ■ POP能弥补OOP-些设计上的不足

回顾OOP

■ OOP的三大特性:封装、继承、多态
■ 继承的经典使用场合

  • 当多个类(比如A、B、C类)具有很多共性时,可以将这些共性抽取到一个父类中(比如D类) , 最后A、B、C类继承D类

OOP的不足

■ 但是有些问题,使用OOP并不能很好解决

  • 如何将BVC、AVC的公共方法run抽取出来?
class BVC: UIViewController {
    func run() {
        print("run")
    }
}
class AVC: UIViewController {
    func run() {
        print("run")
    }
}

■ 基于OOP想到的一些解决方案?
1.将run方法放到另一个对象A中,然后BVC、DVC拥有对象A属性

  • 多了一些额外的依赖关系
    2.将run方法增加到UIViewController分类中
  • UIViewController会越来越臃肿,而且会影响它的其他所有子类
    3.将run方法抽取到新的父类,采用多继承? ( C++支持多继承 )
  • 会增加程序设计复杂度,产生菱形继承等问题,需要开发者额外解决

POP的解决方案

protocol Runnable {
    func run()
}
extension Runnable {
    func run() {
        print("run")
    }
}
class AVC: UIViewController, Runnable {}
class BVC: UIViewController, Runnable {}

POP的注意点

■ 优先考虑创建协议,而不是父类(基类)
■ 优先考虑值类型(struct、enum), 而不是引用类型( class )
■ 巧用协议的扩展功能
■ 不要为了面向协议而使用协议

利用协议实现前缀效果

struct HY<Base> {
    let base: Base
    init(_ base: Base) { self.base = base }
}
protocol HYCompatible {}
extension HYCompatible {
    static var hy: HY<Self>.Type {
        get { HY<Self>.self }
        set {}
    }
    var hy: HY<Self> {
        get { HY(self) }
        set {}
    }
}
extension String: HYCompatible {}
extension HY where Base == String {
    var numberCount: Int {
        var count = 0
        for c in base where ("0"..."9").contains(c) {
            count += 1
        }
        return count
    }
    static func test() {
        print("\(Self.self) static")
    }
}
var string = "123qwe434"
// 6
print(string.hy.numberCount)
//HY<String> static
print(String.hy.test())
class Person {}
extension Person: HYCompatible {}
extension HY where Base: Person {
    func run() {
        print("\(self) run")
    }
    static func test() { print("\(Self.self) static func") }
}
class Teacher: Person {}
extension HY where Base == Teacher {
    func teach() {
        print("teach")
    }
}
//HY<Person> static func
Person.hy.test()
//HY<Teacher> static func
Teacher.hy.test()
let p = Person()
//HY<Person>(base: Adam_20220925_SwiftBase.Person) run
p.hy.run()
let t = Teacher()
//HY<Teacher>(base: Adam_20220925_SwiftBase.Teacher) run
t.hy.run()

给NSString加前缀

extension NSString: HYCompatible {}
extension HY where Base: NSString {
    var numberCount: Int {
        var count = 0
        for c in base as String where ("0"..."9").contains(c) {
            count += 1
        }
        return count
    }
}
var string1 = "123XXX"
var string2: NSString = "123XXX"
var string3: NSMutableString = "123XXX"
print(string1.hy.numberCount)
print(string2.hy.numberCount)
print(string3.hy.numberCount)

给NSString加前缀优化

extension String: HYCompatible {}
extension NSString: HYCompatible {}

extension HY where Base: ExpressibleByStringLiteral {
    var numberCount: Int {
        var count = 0
        for c in (base as! String) where ("0"..."9").contains(c) {
            count += 1
        }
        return count
    }
}
var string1 = "123XXX"
var string2: NSString = "1234XXX"
var string3: NSMutableString = "12345XXX"
print(string1.hy.numberCount)
print(string2.hy.numberCount)
print(string3.hy.numberCount)

利用协议判断类型

protocol ArrayType {}
extension Array: ArrayType {}
extension NSArray: ArrayType {}
func isArrayType(_ type: Any.Type) -> Bool {
    type is ArrayType.Type
}
print(isArrayType([Int].self))
print(isArrayType([Double].self))
print(isArrayType([Any].self))
print(isArrayType(NSArray.self))
print(isArrayType(NSMutableArray.self))
print(isArrayType(String.self))

响应式编程

响应式编程( Reactive Programming ,简称RP )

  • 也是y一种编程范式,于1997年提出,可以简化异步编程,提供更优雅的数据绑定
  • 一般与函数式融合在一起,所以也会叫做:函数响应式编程( Functional Reactive Programming,简称FRP )
    比较著名的、成熟的响应式框架

❇️ ReactiveCocoa :

❇️ ReactiveX

RxSwift

■ RxSwift ( ReactiveX for Swift ) , ReactiveX的Swift版本

■ RxSwift的安装
①Podfile
use_ frameworks!
target 'target_ name' do
pod 'RxSwift', '~> 6'
pod 'RxCocoa', '~> 6'
end

②命令行
pod repo update
pod install
③导入模块
import RxSwift
import RxCocoa

■ 模块说明

  • RxSwift : Rx标准API的Swift实现,不包括任何iOS相关的内容
  • RxCocoa :基于RxSwift ,给iOS UI控件扩展了很多Rx特性

RxSwift核心类

  • Observable:负责发送事件(Event)
  • Observer: 负责订阅Observable,监听Observable发送的事件

Event 类型

/// **next\* (error | completed)**
@frozen 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
}

创建、订阅observable1


//        let observable = Observable<Int>.create { observer in
//            observer.onNext(1)
//            observer.onCompleted()
//            return Disposables.create()
//        }
// 等价于
//        let observable = Observable.just(1)
//        let observable = Observable.of(1)
        let observable = Observable.from([1])
        
        observable.subscribe{ event in
            print(event)
        }.dispose()
//        let observable = Observable<Int>.create { observer in
//            observer.onNext(1)
//            observer.onNext(2)
//            observer.onNext(3)
//            observer.onCompleted()
//            return Disposables.create()
//        }
// 等价于
//        let observable = Observable.of(1, 2, 3)
        let observable = Observable.from([1, 2, 3])
        
        observable.subscribe{ event in
            print(event)
        }.dispose()
        
        
        observable.subscribe {
            print("next", $0)
        } onError: {
            print("onError", $0)
        } onCompleted: {
            print("onCompleted")
        } onDisposed: {
            print("onDisposed")
        }.dispose()
        // 等价于
        observable.subscribe (onNext: {
            print("next", $0)
        }, onError: {
            print("onError", $0)
        }, onCompleted: {
            print("onCompleted")
        }, onDisposed: {
            print("onDisposed")
        }).dispose()

创建、订阅observable2

let observable = Observable<Int>.timer(.seconds(3),
                                               period: .seconds(1),
                                               scheduler: MainScheduler.instance)
        observable
            .map{ "数值为\($0)"}
            .bind(to: label.rx.text)
            .disposed(by: bag)

Disposable

    ■每当Observable被订阅时,都会返回一个Disposable实例,当调用Disposable的dispose ,就相当于取消订阅
    ■在不需要再接收事件时,建议取消订阅,释放资源。有3种常见方式取消订阅
       let observable = Observable.from([1, 2, 3])
       //立即取消订阅(一次性订阅)
       observable.subscribe { event in
           print (event )
       }.dispose()
       //当bag销毁(deinit) 时,会自动调用Disposab le实例的dispose
       observable.subscribe { event in
           print(event)
       }.disposed(by: bag)
       // self销毁时(deinit) 时,会自动调用Disposable实例的dispose
       let _ = observable.takeUntil(self.rx.deallocated).subscribe { event in
           print(event)
       }

Binder Label

let binder = Binder(label) { label, value in
            label.text = value
        }
        Observable.just(1).map { "数值为\($0)" }.subscribe(binder).dispose()
        Observable.from([1, 2, 3]).map { "数值为\($0)" }.bind(to: binder).dispose()

Binder UIButton

        let observable = Observable<Int>.timer(.seconds(2),
                                               period: .seconds(1),
                                               scheduler: MainScheduler.instance)
        let binder = Binder<Bool>(button) { button, value in
            button.isHidden = value
        }
        observable.map { $0 % 2 == 0}.bind(to: binder).disposed(by: bag)

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

推荐阅读更多精彩内容