RxSwift
(github.com/devxoul)
• StyleShare Inc.
• SW Maestro 2
• RxSwift
RxSwift?
RxSwift = ReactiveX + Swift
RxSwift = ReactiveX + Swift
RxSwift = ReactiveX + Swift
?
RxSwift = ReactiveX + Swift
Re + ActiveX
RxSwift = ReactiveX + Swift
Reactive + Extension
RxSwift = ReactiveX + Swift
RxSwift = ReactiveX + Swift
?
a = 10
b = a * 2
a = 10
b = a * 2
print(b)
a = 10
b = a * 2
print(b) // 20
a = 10
b = a * 2
print(b) // 20
a = 30
a = 10
b = a * 2
print(b) // 20
a = 30
print(b)
a = 10
b = a * 2
print(b) // 20
a = 30
print(b) // 20
a = 10
b = a * 2
print(b) // 20
a = 30
print(b) // 20 60
ReactiveX
Observer
Pattern
Iterator
Pattern
Functional
Programming
ReactiveX
Observer
Pattern
Iterator
Pattern
Functional
Programming
Observable
Observer
Observable
Observer
observe
Observable
Observer
observe notify
Observable
ObserverObserver Observer
ReactiveX
Observer
Pattern
Iterator
Pattern
Functional
Programming
Iterator next()
hasNext()
Iterator next()
hasNext()
10 20 30 40
Iterator next()
hasNext()
10 20 30 40
Iterator next()
hasNext()
10 20 30 40
iterator.next()
Iterator next()
hasNext()
10 20 30 40
iterator.next() // 10
Iterator next()
hasNext()
10 20 30 40
Iterator next()
hasNext()
10 20 30 40
iterator.next()
Iterator next()
hasNext()
10 20 30 40
iterator.next() // 20
Iterator next()
hasNext()
10 20 30 40
Iterator next()
hasNext()
10 20 30 40
iterator.next()
Iterator next()
hasNext()
10 20 30 40
iterator.next() // 30
Iterator next()
hasNext()
10 20 30 40
Iterator next()
hasNext()
10 20 30 40
iterator.next()
Iterator next()
hasNext()
10 20 30 40
iterator.next() // 40
Iterator next()
hasNext()
10 20 30 40
Iterator next()
hasNext()
10 20 30 40
iterator.hasNext() // false
ReactiveX
Observer
Pattern
Iterator
Pattern
Functional
Programming
•
• ,
•
•
• ...
higher-order fuction
pure fuction
•
•
• ...
higher-order fuction
pure fuction
1.
2.
var array = [2, 6, 3, 1, 7]
array.sort(by: { a, b in
return a < b
})
var array = [2, 6, 3, 1, 7]
array.sort(by: { a, b in
return a < b
})
var array = [2, 6, 3, 1, 7]
array.sort(by: { a, b in
return a < b
})
•
•
• ...
higher-order fuction
pure fuction
•
•
•
•
( , , I/O )
•
•
var rate = 1130
func krw(usd: Int) -> Int {
return usd * rate
}
var rate = 1130
func krw(usd: Int) -> Int {
return usd * rate
}
krw(usd: 2) // 2260
krw(usd: 3) // 3390
var rate = 1130
func krw(usd: Int) -> Int {
return usd * rate
}
rate = 1140
krw(usd: 2) // 2260
krw(usd: 3) // 3390
var rate = 1130
func krw(usd: Int) -> Int {
return usd * rate
}
rate = 1140
krw(usd: 2) // 2260 2280
krw(usd: 3) // 3390 3420
var rate = 1130
func krw(usd: Int) -> Int {
return usd * rate
}
rate = 1140
krw(usd: 2) // 2260 2280
krw(usd: 3) // 3390 3420
var rate = 1130
func krw(usd: Int) -> Int {
return usd * rate
}
rate = 1140
krw(usd: 2) // 2260 2280
krw(usd: 3) // 3390 3420
func krw(usd: Int) -> Int {
return usd * rate
}
func krw(usd: Int ) -> Int {
return usd * rate
}
func krw(usd: Int, rate: Int) -> Int {
return usd * rate
}
func krw(usd: Int, rate: Int) -> Int {
return usd * rate
}
ReactiveX
Observable
Pattern
Iterator
Pattern
Functional
Programming
a 10
b a * 2
a 10
b 20
a
b
10 20 30 40a
20 40 60 80b
10 20 30 40a
20 40 60 80b
10 20 30 40a
20 40 60 80b
map(x => x * 2)
10 20 30 40a
20 40 60 80b
map(x => x * 2)
Marble Diagrams
20 40 60 80b
map(x => x * 2)
Observable.from([10, 20, 30, 40])
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value)
})
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value)
})
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value)
})
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value) // 20
})
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value)
})
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value)
})
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value) // 40
})
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value)
})
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value)
})
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value) // 60
})
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value)
})
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value)
})
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value) // 80
})
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value) // 80
})
Observable
Observable.from([10, 20, 30, 40])
.map { $0 * 2 }
.subscribe(onNext: { value in
print(value) // 80
})
Observer
Password
(4 )
1
(4 )
. ❌
12
(4 )
. ❌
123
(4 )
. ❌
1234
(4 )
. 👍
"1"
"1"
map(str => str.length >= 4)
"1"
map(str => str.length >= 4)
false
"1"
false
map(str => str.length >= 4)
map(valid => getMessage(valid))
"1"
false
map(str => str.length >= 4)
"❌ "
map(valid => getMessage(valid))
"1" "12"
false
map(str => str.length >= 4)
"❌ "
map(valid => getMessage(valid))
"1" "12"
false false
map(str => str.length >= 4)
"❌ "
map(valid => getMessage(valid))
"1" "12"
false false
map(str => str.length >= 4)
"❌ " "❌ "
map(valid => getMessage(valid))
"1" "12" "123"
false false
map(str => str.length >= 4)
"❌ " "❌ "
map(valid => getMessage(valid))
"1" "12" "123"
false false false
map(str => str.length >= 4)
"❌ " "❌ "
map(valid => getMessage(valid))
"1" "12" "123"
false false false
map(str => str.length >= 4)
"❌ " "❌ " "❌ "
map(valid => getMessage(valid))
"1" "12" "123" "1234"
false false false
map(str => str.length >= 4)
"❌ " "❌ " "❌ "
map(valid => getMessage(valid))
"1" "12" "123" "1234"
false false false true
map(str => str.length >= 4)
"❌ " "❌ " "❌ "
map(valid => getMessage(valid))
false false false true
map(str => str.length >= 4)
map(valid => getMessage(valid))
"1" "12" "123" "1234"
"❌ " "❌ " "❌ " "👍"
passwordField.rx.text.orEmpty
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
}
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
"1"
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
false
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
"12"
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
false
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
"123"
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
false
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
"1234"
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
true
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
passwordField.rx.text.orEmpty
.map { $0.characters.count >= 4 }
.map { isValid in
if isValid {
return " . 👍"
} else {
return " . ❌"
}
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.rx.text = message
})
: Observable<String>
: Observable<Int>
: Observable<Void>
: Observable<CGPoint>
...
Observable
"1" "12" "123" "1234"
"1"
"1"
"1" "12"
"1" "12"
"1" "12" "123"
"1" "12" "123"
"1" "12" "123" "1234"
Nickname
. ❌
. ❌
. 👍
API
API
=
nicknameField.rx.text.orEmpty
nicknameField.rx.text.orEmpty
.map { nickname -> Bool in
// return true of false
}
🤔 ?
nicknameField.rx.text.orEmpty
.map { nickname -> Bool in
// return true of false
}
nicknameField.rx.text.orEmpty
. map { nickname -> Bool in
}
nicknameField.rx.text.orEmpty
.flatMap { nickname -> Observable<Bool> in
}
nicknameField.rx.text.orEmpty
.flatMap { nickname -> Observable<Bool> in
return API.checkNickname(nickname)
}
nicknameField.rx.text.orEmpty
.flatMap { nickname -> Observable<Bool> in
return API.checkNickname(nickname)
}
.map { isAvailable in
return getMessage(isAvailable)
}
nicknameField.rx.text.orEmpty
.flatMap { nickname -> Observable<Bool> in
return API.checkNickname(nickname)
}
.map { isAvailable in
return getMessage(isAvailable)
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.text = message
})
map() vs flatMap()
20 40 60 80
map(x => x * 2)
10 20 30 40
map() vs flatMap()
flatMap( => )
map() vs flatMap()
flatMap( => )
map() vs flatMap()
flatMap( => )
map() vs flatMap()
flatMap( => )
map() vs flatMap()
flatMap( => )
map() vs flatMap()
flatMap( => )
map() vs flatMap()
flatMap( => )
map() vs flatMap()
flatMap( => )
map() vs flatMap()
flatMap( => )
map() vs flatMap()
flatMap( => )
map() vs flatMap()
flatMap( => )
Nickname
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
nicknameField.rx.text.orEmpty
.flatMap { nickname -> Observable<Bool> in
return API.checkNickname(nickname)
}
.map { isAvailable in
return getMessage(isAvailable)
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.text = message
})
nicknameField.rx.text.orEmpty
.debounce(0.3, scheduler: MainScheduler.instance)
.flatMap { nickname -> Observable<Bool> in
return API.checkNickname(nickname)
}
.map { isAvailable in
return getMessage(isAvailable)
}
.subscribe(onNext: { [weak self] message in
self?.messageLabel.text = message
})
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
0.3s
0.3s
0.3s
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
checkNickname(" ")
0.3s
0.3s
0.3s
1 4
debounce
1 2 3 4
, Observable
Timer
00:17
Timer
00:17
Observable<Int>
.interval(1, scheduler: ...)
.subscribe(onNext: { tick in
print(tick)
})
Timer
00:17
Start
17
18
19
20
21
...
Start
Start
Observable
Timer
00:17
Observable<Int>
.interval(1, scheduler: ...)
.subscribe(onNext: { tick in
print(tick)
})
Timer
00:17
var disposeBag = DisposeBag()
Observable<Int>
.interval(1, scheduler: ...)
.subscribe(onNext: { tick in
print(tick)
})
.disposed(by: disposeBag)
Timer
00:17
var disposeBag = DisposeBag()
Observable<Int>
.interval(1, scheduler: ...)
.subscribe(onNext: { tick in
print(tick)
})
.disposed(by: disposeBag)
var disposeBag = DisposeBag()
Observable<Int>
.interval(1, scheduler: ...)
.subscribe(onNext: { tick in
print(tick)
})
.disposed(by: disposeBag)
Start
var disposeBag = DisposeBag()
Observable<Int>
.interval(1, scheduler: ...)
.subscribe(onNext: { tick in
print(tick)
})
.disposed(by: disposeBag)
Start
var disposeBag = DisposeBag()
Observable<Int>
.interval(1, scheduler: ...)
.subscribe(onNext: { tick in
print(tick)
})
.disposed(by: disposeBag)
Start
var disposeBag = DisposeBag()
Observable<Int>
.interval(1, scheduler: ...)
.subscribe(onNext: { tick in
print(tick)
})
.disposed(by: disposeBag)
Start
Dispose
•
•
•
RxSwift
http://reactivex.io
http://rxswift.org
http://community.rxswift.org
https://rxswift-slack.herokuapp.com
.
(github.com/devxoul)

RxSwift 시작하기