【RxSwift】Hot変換オペレータの1つであるshareReplayを使ってみる

公開日: : iPhoneアプリ開発

20160423-124300.jpg

最近、RxSwiftについて勉強中です。

今回は、Hot変換オペレータの1つであるshareReplayを使ってみたので記事にまとめてみました。

RxSwiftはまだ勉強中なので、どこか間違っている箇所があればご指摘いただけると嬉しいです。 ⇒ https://twitter.com/akio0911

    

スポンサード リンク

UITextFieldの入力をそのまま2つのUILabelに表示する

例として、1つのUITextFieldに入力された文字列を、2つのラベルにそのまま表示するアプリを考えてみましょう。

RxSwiftを使ったコードとしては、以下のようになります。

@IBOutlet weak var textField: UITextField!

@IBOutlet weak var label1: UILabel!
@IBOutlet weak var label2: UILabel!

let disposeBag = DisposeBag()

override func viewDidLoad() {
    super.viewDidLoad()

    textField.rx_text.bindTo(label1.rx_text).addDisposableTo(disposeBag)
    textField.rx_text.bindTo(label2.rx_text).addDisposableTo(disposeBag)
}

これでユーザーがテキストフィールドに文字列を入力すると、ラベルへリアルタイムに反映されるようになります。

ここまでは特に問題ないかなと思います。

    

UITextFieldの入力を加工して、2つのUILabelに表示する

次はそのまま表示せずに簡単な加工を行ってみましょう。

テキストフィールドに入力された文字列の先頭と末尾に*を付加し、それを2つのラベルへリアルタイムに反映させます。

コードは以下のようになります。

let num = textField.rx_text.map{ text -> String in
    print("map \(text)")
    return "*\(text)*"
}

num.bindTo(label1.rx_text).addDisposableTo(disposeBag)
num.bindTo(label2.rx_text).addDisposableTo(disposeBag)

    

さて、このコードを早速実行してみましょう。

map 
map 
map 1
map 1
map 12
map 12

なんと、文字に変更を加えるたび、mapのクロージャー内の処理が2回ずつ実行されてしまっています。

今回の場合だと同じ加工を施した同じ内容を2つのラベルに表示させたいので、mapのクロージャー内の処理は1変更に対して1回だけ実行されればいいのですが・・・。

    

20160423-123905.jpg

なぜこのようなことが起こってしまったのかというと、2つのオブジェクトにbindしsubscribeすると、別々の2本のストリームが生成されてしまうんだそうです。

このような性質を持つものを、Cold Observableと呼ぶようです。

参考 : RxのHotとColdについて – Qiita

今回のような軽い処理であれば大きな問題にはなりませんけど、重い処理だったり、ファイルIOや通信処理などが含まれていたりすると、それらが2回ずつ実行されると困りますよね。

    

shareReplayを使ってみる

ではどうすれば良いのかというと、Coldと呼ぶくらいですから、Hot Observableを使います。

Cold ObservableをHot Observableに変換してくれる「Hot変換オペレータ」というものがあるので、その中の1つである「shareReplay」を使います。

let num = textField.rx_text
    .map{ text -> String in
        print("map \(text)")
        return "*\(text)*"
    }
    .shareReplay(1)

num.bindTo(label1.rx_text).addDisposableTo(disposeBag)
num.bindTo(label2.rx_text).addDisposableTo(disposeBag)

shareReplay(1)は、内部的にはreplay(1).refCountを呼び出してるだけのショートカットなんだそうですが、詳細についてはまた別記事を書こうと思います。

参考 : [RxSwift] shareReplayをちゃんと書いてお行儀良くストリームを購読しよう – Qiita

    

実行結果は以下のようになります。

map 
map 1
map 12
map 123

テキストフィールドへの1変更に対して、mapのクロージャー内の処理が1回ずつ呼ばれています。

    

20160423-123910.jpg

これは、Hot Observerの「ストリームを分岐する性質」によるものだそうです。

    

@akio0911はこう思った。

Cold ObservableとHot Observableの違いが分かっていないと、意図せず無駄な処理が発生してしまいそうなので、ここは注意が必要ですね。

    

    

この記事が気に入ったら「いいね!」しよう

follow us in feedly

Feedlyで最新記事を購読

Twitterで更新情報をゲット!

LINEでご感想・ご要望お送りください!
(スマホでLINEを起動 > 友だち追加 > QRコード)

関連記事

20141006-103526.jpg

Xcode 6におけるNSUserDefaultsのplistの在り処と、内容を確認・書き換え・削除する方法

Xcode 6にて、NSUserDefaultsのplistファイルの在り処と、内容を確認・書き

記事を読む

I20160130-110811.jpg

【解決】Redundant conformance of ‘NSDate’ to protocol ‘Comparable’エラーが出た時の対処方法

アプリ開発中に「Redundant conformance of 'NSDate' to pro

記事を読む

I20150802-085831.jpg

Swift 1.2でtouchesBeganメソッドのNSSetがSetへ変更されました

Swift 1.2で、touchesBegan・touchesMoved・touchesCanc

記事を読む

I20151102-201945.jpg

Swift 2の文法を学べるオススメ本「Swift 2標準ガイドブック」

Swift 2の文法について学べる書籍「Swift 2標準ガイドブック」が発売されたので、紹介し

記事を読む

I20150714-010114.jpg

【Swift】Objective-Cで書いたクラスをメソッド単位で少しずつSwiftへ移行する方法

Objective-Cで作った既存のアプリをSwiftで書き直したいと思っているのですが、一

記事を読む

I20150805-182005.jpg

Swiftにおける関数型のエッセンスを学ぶために「関数プログラミング実践入門」を読み始めた

Swiftの関数型言語的な特徴についての話を各所で見るようになりましたが、自分は関数型言語の

記事を読む

20150525-224340.jpg

【Swift】クロージャをメソッドや関数の引数として渡す方法

Swiftでメソッドの引数としてクロージャを渡す方法について解説したいと思います。     

記事を読む

I20160119-155700.jpg

Xcodeで80文字目のところにガイドラインを表示する方法

Xcodeで80文字目のところに線(ガイドライン)を表示する方法について解説します! 「ソース

記事を読む

20141228-232009.jpg

【Swift】as?演算子を使ったキャストでは結果がoptionalとなる

今日はSwiftのas演算子とas?演算子について解説したいと思います。as演算子について

記事を読む

I20160228-162014.jpg

詳解 Swift 改訂版のKindle版が発売。リフローに対応!

今日気付いたんですが、「詳解 Swift 改訂版」のKindle版がいつの間にか発売されていまし

記事を読む

I20170228-002742.jpg
約3ヶ月で体脂肪率を6.2%も落とせたキッカケについて

(右上の赤枠内がダイエット開始前、左下の赤枠内が3ヶ月後の数値

I20161224-174949.jpg
「季節の野菜を直接配送!季節のスムージー」を買ってみた

「FiNCモール」で、「季節の野菜を直接配送!季節のスムージー」を

I20161002-152537.jpg
【メンズネイル】東京・新宿のネイルサロンでマットネイルしてもらった

2016年7月18日、東京・新宿のメンズOKなネイルサロン「Tot

I20160925-163452.jpg
タブバーアイコン非選択時の色を変更する方法【iOS 10】

UITabBarControllerで、タブバーアイコン非選択時の

I20160924-123726.jpg
iPad Pro 9.7インチ SIMフリーモデルにFREETELのSIMを入れて使ってみた

今までiPad miniで使っていたFREETELのSIM

→もっと見る

PAGE TOP ↑