質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.20%

dataSourceの意味合いがよくわりません。

解決済

回答 2

投稿

  • 評価
  • クリップ 1
  • VIEW 302

magiee

score 28

iPhoneアプリ開発集中講座の本で分からないところがあります。

タイマーアプリを作成する為に、PickerViewを利用しています。
その中で設定画面起動時に秒数をセットする項目があります。それが以下のコード。

    override func viewDidLoad() {
        super.viewDidLoad()

        timerSettingPicker.delegate = self
        timerSettingPicker.dataSource = self

        let settings = UserDefaults.standard
        let timerValue = settings.integer(forKey: settingKey)

        for row in 0..<settingArray.count {
            if settingArray[row] == timerValue {
                timerSettingPicker.selectRow(row, inComponent: 0, animated: true)
            }
        }
    }

この中のtimerSettingPicker.dataSource = self の役割がかりません。
timerSettingPicker.dataSource = self のコードを削除しても、問題なく秒数がセットされカウントされました。
このコードはどのような役割をしているのでしょうか。ご教授のほどよろしくお願いします。

ソースコード全文

import UIKit

class SettingViewController: UIViewController, UIPickerViewDataSource, UIPickerViewDelegate {

    let settingArray : [Int] = [5,10,20,30,40,50,60]

    let settingKey = "timer_value"


    //viewDidLoadメソッドは、SettingViewControllerの起動時に一度だけ実行されるメソッド
    override func viewDidLoad() {
        super.viewDidLoad()


        timerSettingPicker.delegate = self //self→自分自身のインスタンスを指定する意味
        timerSettingPicker.dataSource = self

        let settings = UserDefaults.standard
        let timerValue = settings.integer(forKey: settingKey)

        for row in 0..<settingArray.count {
            if settingArray[row] == timerValue {
                timerSettingPicker.selectRow(row, inComponent: 0, animated: true)
            }
        }
    }


    /*
    // MARK: - Navigation

    // In a storyboard-based application, you will often want to do a little preparation before navigation
    override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
        // Get the new view controller using segue.destination.
        // Pass the selected object to the new view controller.
    }
    */

    @IBOutlet weak var timerSettingPicker: UIPickerView!



    @IBAction func decisionButtonAction(_ sender: Any) {
        //前画面に戻る
        _ = navigationController?.popViewController(animated: true)
    }

    //秒数設定の列数を決定。今回は1列の為、1
    func numberOfComponents(in pickerView: UIPickerView) -> Int {
        return 1
    }

    //秒数設定の行数を決定。settingArrayから取得する為、settingArrayの配列によって行数が変わる
    func pickerView(_ pickerView: UIPickerView, numberOfRowsInComponent component: Int) -> Int {
        return settingArray.count
    }

    //UIPickerViewの表示する内容を設定 秒数が全て?になる。
    func pickerView(_ pickerView: UIPickerView, titleForRow row: Int, forComponent component: Int) -> String? {
        return String(settingArray[row])
    }

    //picker選択時に実行
    func pickerView(_ pickerView: UIPickerView, didSelectRow row: Int, inComponent component: Int) {
        // UserDefaultsの設定
        let settings = UserDefaults.standard //iosアプリ内にデータを保存する方法の一つ
        settings.setValue(settingArray[row], forKey: settingKey) //選択された秒数を保存している。20秒で設定しても、デフォルトの10秒のまま。
        settings.synchronize() //現状不要(メソッドを呼び出さないと保存されない為、必要だった)
    }

}

使用している言語

swift
Xvode 11.4

ご教授のほど、よろしくお願い致します。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 2

checkベストアンサー

+1

dataSourceそのものの意味については、Tetsukickさんが書かれている通りです。

一方、ご質問の「dataSourceの設定を削除しても動く理由」ですが、それを裏付けるような明確なドキュメントなどは私が調べた限り見つかりませんでした。いくつかの条件下では設定していなくても動くようです。

実験的なコードを書いて試した範囲で分かったことは

  • dataSourceが設定されていない場合(値がnilの場合)は、delegateで設定されたクラスを参照する
  • delegatenilの場合はそもそも動かない

ということ程度でした。

ただし、基本としてはdataSourceを設定するということに落ち着くと思いますから、いまはそれ以上深く考えないほうがいいと思います。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2020/04/30 08:59

    回答していただきありがとうございます。
    あまり必要とはされていないが、DataSourceとDelegateの両方を記載していた方が、とりあえず無難にいけますよっといいう事なんですね。

    キャンセル

  • 2020/04/30 09:31

    いや、必須です。
    たまたまdataSourceを記述しなくとも動いただけで、今後同じような振る舞いをする保証はありません。
    なので、両方とも記述する必要はありますね。

    キャンセル

  • 2020/04/30 20:30

    必須なのですね。分かりました、ありがとうございます。

    キャンセル

0

DataSource とは一言で言うと「Picker内のテーブルに表示する内容を提供するもの」

です。

selfを渡しているのは、
SettingViewController が UIPickerViewDataSource を継承しているからです。

実際に以下のメソッドがUIPickerViewDataSourceに定義されているメソッドです。
Picker内の内容を描画する際に呼び出されます。

    //UIPickerViewの表示する内容を設定 秒数が全て?になる。
    func pickerView(_ pickerView: UIPickerView, titleForRow row: Int, forComponent component: Int) -> String? {
        return String(settingArray[row])
    }

参考まで
http://sayulemon46.hatenablog.com/entry/2017/03/06/171934

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2020/04/30 08:56

    回答していただきありがとうございます。
    DataSourceとDelegateの分かりやすい資料も頂き、参考になりました。
    ありがとうございます。

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.20%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る