swift3 デリゲートメソッドの実装について

デリゲートメソッドをどのように実際に利用するか説明

呼び出し元    

 

//会話履歴テーブル,デリゲートで呼ぶ

var delegate: ConversationDelegate?

delegate?.ConversationSelected(Conversation: datMeeting!)

//datMeetingは実引数

 

 

// 会話履歴プロトコル

protocol ConversationDelegate {

    func ConversationSelected( Conversation: DatMeeting)    

}

 

//実装先 呼び出されている方

   func ConversationSelected(Conversation: DatMeeting) {

        //ここでdatperson id

        perName.text = datPerson.name

        //少配布物を作成

        distSmall.text = person.distSmall

        convCont.text = Conversation.content

    }

 

主に画面の左右に別れているときに利用した。

画面が変わると全部メモリでは落ちている。

そこで次の画面につなげる仕組みの一つがデリゲート。

 

他の例 これはデリゲートメソッドではないけど

Msperson画面が2つに割れているもの、詳細 MAP

  private weak var msPersonData: MsPersonDataPad!

 

  override func prepare(for segue: UIStoryboardSegue, sender: Any?) {

        super.prepare(for: segue, sender: sender)

        if segue.destination is MsPersonDataPad {

            // Embeded segue 受信のタイミングで MsPersonData インスタンスを取得し

            // 編集モードとデータを設定しておく

            msPersonData = segue.destination as! MsPersonDataPad

            msPersonData.setupDatPerson(.New, datPerson: datPerson)

        }

    }

 

    /// 編集モードでのdatParson(訪問先データ)の運用を開始する

    ///

    /// .Empty又は.Newの場合、datPerson は nil(default)、datPerson != nil の場合は

    /// 指定 mode によらず.Existing になる。

    /// - Parameters:

    ///   - mode: 編集モード

    ///   - datPerson: 訪問先データ

    func setupDatPerson(_ mode: EditMode, datPerson: DatPerson? = nil) {

        self.editMode =

            datPerson != nil ? .Existing : mode == .New ? .New : .Empty

        datPersonImpl =

            self.editMode != .New ? datPerson : DatPerson.newFormalOne()

        // viewがすでに動作している(viewWillAppear <= state < dismiss)

        if canDisplyData {

            setDataOnUIs()

        }

    }

 

   /// 編集モードを表す

    ///

    /// - Empty: 該当無し(検索と訪問先の画面で検索結果が空,...等)

    /// - New: 新規登録

    /// - Existing: 既存訪問先(詳細表示と編集)

    enum EditMode: Int {

        case Empty

        case New

        case Existing

    }

 

 

SWIFT3 クロージャーについて

まずクロージャーについて

クロージャーとはなんなのか?

という質問について、まず

関数もクロージャーの一種である。

という説明がもっともわかりやすいと思われる。

関数は、処理をひとまとめにしたものだが、クロージャーより、詳細に設定できると考えると良い。

全く別物ではないのである。

基本形は

 

{ (引数名1:型,引数名2:型...) -> 戻り値の型 in 

クロージャーの実行時に実行される文

必要に応じてreturn で戻り値を返却

}

 

大外は{}で囲われる。inがポイントになるので赤字にした。

in キーワードで引数と処理が区別される。

 

関数と違い

クロージャーでは、外部引数名は利用できない。

デフォルト引数も利用できない。

通常のfuncに付け加えて利用できるものは、

簡略引数名が利用できる。 $0,$1など。$にインデックスをつけたものが利用される。

(例)関数

//2つの引数の和を戻す関数

func B(x:Int,y:Int)->Int{

    return x + y

}

print(B(x: 100, y: 500))

これが普通の関数

 

(例)クロージャー

var b = { (x:Int,y:Int) -> Int in return x + y}

print(b(200, 200))

こうなる

関数名がないので、他から呼び出しはできない。

 

//クロージャで型を宣言

var b_1 : (Int,Int) -> Int  //b_1はint型2つを引数に取るし、戻り値はint型を取りますということ

b_1 = { (x,y) -> Int in return x + y}

//{}がクロージャーで,x,yの引数を渡したら足し算して返すだけ

print(b_1(100,20))

 =>120

 

そもそも、なぜクロージャーがあるのか?

というと、関数を変数や、定数に代入したり、別の関数の引数に渡したりできるから。

これ大切ですね。

関数を変数、定数に代入するときは、関数名だけの式で関数を参照します。

 

 

関数内関数は禁止されています。エラーになります。

そこでクロージャーを利用します。

func aho (){

   func hoge(){

   }

}  //エラー

こんなふうに、関数内関数は利用できません。

でもクロージャーなら可能です。

(省略、想像してください。)

 

次に関数の引数となるクロージャーを関数として定義することを見てみてたい。

let aho = [1,2,3]

let doubleAho = aho.map{ $0 * 2 }

//aho配列の中から要素を一つ一つ取り出して$0に代入しようね。そしたら、それを2倍してね!ちなみにmap関数は配列に対して操作する関数です。こちらを参照ください。

doubleAho // [2,4,6]

 

関数をクロージャーとして扱う

func double(_ x: Int) -> Int{

    return x * 2

}

let aho = [1,2,3]

let ddoubleAho = aho.map(double)  //ここで関数をクロージャーとして呼び出しています

doubleAho // [2,4,6]

こうするとどのような操作をしているか、関数の中ではっきりと分かりますね。

 

 クロージャーを利用した変数、定数の初期化

クロージャーを利用すると何をしているかわかりやすい。

2次元配列を例に取る。//縦と横に配列がある、エクセルのような状態になっているもの。

var aho =[ [1,1,1],[1,1,1],[1,1,1] ]

aho => [ [1,1,1],[1,1,1],[1,1,1] ]

全部入力しているので、2次元配列を生成してみる

 

var aho = Array(repeating: Array(repeating: 1, count: 3), count: 3)

//ちなみにrepeatingとは配列を初期化するメソッド 

    ///     let fiveZs = Array(repeating: "Z", count: 5)

    ///     print(fiveZs)

    ///     // Prints "["Z", "Z", "Z", "Z", "Z"]"

    ///

    /// - Parameters:

    ///   - repeating 繰り返すべき要素

    ///   - count: 繰り返す回数、0以上

この中では1を入れて3回繰り返してね。

更にその外では、[1,1,1]をいれて3回繰り返してね

というメソッドである。

 

aho => [ [1,1,1],[1,1,1],[1,1,1] ]

 

これをクロージャで表現すると

var aho: [ [Int] ] = {

 let kurikaeshi_times = 3

 let low = Array(repeating: 1, count: kurikaeshi_times)

 let excel = Array(repeating: row, count: kurikaeshi_times

 return excel

}

aho => [ [1,1,1],[1,1,1],[1,1,1] ]

クロージャはわかりやすいですね。

 

補足するとクロージャーはコールバックとして利用されることが多いと思う。

非同期処理のためのDispatchモジュールはほとんど、コールバックを引数で受け取る。

MAPの表示なども引数をクロージャーで渡す。

クロージャーは、引数として利用することが多いと思う。