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

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

新規登録して質問してみよう
ただいま回答率
85.48%
Swift

Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

Q&A

解決済

2回答

274閲覧

structの使い方、呼ばれ方(Swift)

退会済みユーザー

退会済みユーザー

総合スコア0

Swift

Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

0グッド

1クリップ

投稿2018/10/23 10:01

structを利用したコードを書いていたのですが、想定通りに動かなかったので質問します。

まず、structを以下のように書きました。

swift

1struct Item { 2 let name: String! 3 4 static let items: [Item] = { 5 // デバッグ用(毎回実行されていない?) 6 print("called") 7 8 return [ 9 Item( 10 name: "test1" 11 ), 12 Item( 13 name: UserDefaults.standard.bool(forKey: "is_true") ? "true" : "false" 14 ) 15 ] 16 }() 17}

Item.itemsをコールするたびにlet items~以下に記載したコードが実行されることを想定して、
コール時のUserDefaultsに保存されている値に応じてvalueを変えたいと思っていました。

そこで別ファイルから以下のようにコールします。
そうすると2回目以降でデバッグ用に仕込んだprintログが出ていないことに気付きました。

swift

1// 1回目 calledが出力される 2Item.items 3 4// UserDefaultに仕込んだ値を変更する処理が入る想定 5 6// 2回目 calledが出力されない 7Item.items

そもそもstructをこのように使うというのが間違っている気もしているのですが、このような現象(Item.itemsを使っても実際のコードが毎回実行されるわけではないように見える)はどうして発生するのでしょうか?

Swiftの基本がわかっていない初心者の質問で恐縮ですが、教えてください。

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

structに限らずですが、

var something: Int = 1

とした場合、somethingに新たに代入を行わない限り、somethingの値は変わりません。 これは当たり前ですね。

そして次のコードは上のコードと結果として同じものとなります。

var something: Int = { return 1 }()

これはクロージャの実行結果を使ってsomethingを初期化しているにすぎません。
分解するとこうです。

let closure = { () -> Int in return 1 } let initialValue: Int = closure() // クロージャを実行し値を取得 var somethinig: Int = initialValue // 取得した値で変数を初期化

つまり、クロージャは初期化時に一度だけ実行されるのみで、変数/定数を何度参照しても、値は変わりません。

投稿2018/10/24 00:09

MasakiHori

総合スコア3384

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

0

ベストアンサー

あまり意図がつかめていないのですが、
https://medium.com/swift-column/singleton-398078bcc58d
↑こちらを読んでほしいです。static let だとシングルトン、しかもこの場合はstructなのでシングルトンもどきが生成されてしまっています。それでも今回のコード { ... }()の中はプロパティの初期化処理なので、2回目以降呼ばれないのは仕様となります。ひとまずプロパティでなくメソッドにすれば毎回実行されますが、あまりよろしくないですね。

Swift

1struct Item { 2 let name: String! 3  4 static func items() -> [Item] { 5 // デバッグ用(毎回実行されていない?) 6 print("called") 7 8 return [ 9 Item( 10 name: "test1" 11 ), 12 Item( 13 name: UserDefaults.standard.bool(forKey: "is_true") ? "true" : "false" 14 ) 15 ] 16 } 17} 18 19Item.items() 20 21Item.items()

投稿2018/10/23 15:52

idonotknow

総合スコア74

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問