Upgrade to Pro
— share decks privately, control downloads, hide ads and more …
Speaker Deck
Features
Speaker Deck
PRO
Sign in
Sign up for free
Search
Search
【いまさら聞けない】SwiftのOptionalってなに?
Search
FromAtom
June 23, 2017
Technology
0
3.5k
【いまさら聞けない】SwiftのOptionalってなに?
社内の勉強会で話しました
FromAtom
June 23, 2017
Tweet
Share
More Decks by FromAtom
See All by FromAtom
「UIは英語なのにアプリ内リンクは日本語だ!」を防ぐコツ / pixiv App Night 2024-10-24
fromatom
0
82
『SwiftUIならiOS, macOSの両方で動くエディタアプリが簡単に作れる』 と思ったら大間違いだよ! / pixiv App Night 2024-01-25
fromatom
1
540
君だけのGFMエディタを作ろう! / iOSDC Japan 2023
fromatom
0
2.1k
サポートiOSバージョンを定期的にあげる仕組みづくり / iOSDC Japan 2022
fromatom
1
3.8k
僕たちが 『Appのプライバシーに関する質問への回答』 そして『ATT』に対応するまでの物語 / iOSDC Japan 2021
fromatom
1
4k
やってみよう! iOSDCデザインスポンサー! / iOSDC Japan 2021 LT
fromatom
2
1.4k
デバッグメニューのメンテナンスが大変だったので、専用アプリを作りました。 / iOSDC Japan 2020
fromatom
9
6.5k
スクリーン配信機能の実装が大変だったので知見をお伝えします / iOSDC2019
fromatom
7
13k
🎉 esa 生誕5周年記念パーティー(\( ⁰⊖⁰)/) 🎉 / esa the 5th anniversary
fromatom
1
4.4k
Other Decks in Technology
See All in Technology
7月のガバクラ利用料が高かったので調べてみた
techniczna
3
800
Snowflakeの生成AI機能を活用したデータ分析アプリの作成 〜Cortex AnalystとCortex Searchの活用とStreamlitアプリでの利用〜
nayuts
0
120
個人CLAUDE.md紹介と設定から学んだこと/introduce-my-claude-md
shibayu36
0
140
Grafana MCPサーバーによるAIエージェント経由でのGrafanaダッシュボード動的生成
hamadakoji
1
900
PRDの正しい使い方 ~AI時代にも効く思考・対話・成長ツールとして~
techtekt
PRO
0
230
ここ一年のCCoEとしてのAWSコスト最適化を振り返る / CCoE AWS Cost Optimization devio2025
masahirokawahara
1
940
生成AI時代のデータ基盤設計〜ペースレイヤリングで実現する高速開発と持続性〜 / Levtech Meetup_Session_2
sansan_randd
1
100
ヘブンバーンズレッドにおける、世界観を活かしたミニゲーム企画の作り方
gree_tech
PRO
0
410
サンドボックス技術でAI利活用を促進する
koh_naga
0
120
「魔法少女まどか☆マギカ Magia Exedra」のグローバル展開を支える、開発チームと翻訳チームの「意識しない協創」を実現するローカライズシステム
gree_tech
PRO
0
420
「魔法少女まどか☆マギカ Magia Exedra」の必殺技演出を徹底解剖! -キャラクターの魅力を最大限にファンに届けるためのこだわり-
gree_tech
PRO
0
420
サポートエンジニアから見たRancher運用の現場
masap
0
110
Featured
See All Featured
Building a Scalable Design System with Sketch
lauravandoore
462
33k
Design and Strategy: How to Deal with People Who Don’t "Get" Design
morganepeng
131
19k
Fireside Chat
paigeccino
39
3.6k
Stop Working from a Prison Cell
hatefulcrawdad
271
21k
Distributed Sagas: A Protocol for Coordinating Microservices
caitiem20
333
22k
Fight the Zombie Pattern Library - RWD Summit 2016
marcelosomers
234
17k
Practical Tips for Bootstrapping Information Extraction Pipelines
honnibal
PRO
23
1.4k
Faster Mobile Websites
deanohume
309
31k
"I'm Feeling Lucky" - Building Great Search Experiences for Today's Users (#IAC19)
danielanewman
229
22k
A better future with KSS
kneath
239
17k
Statistics for Hackers
jakevdp
799
220k
Refactoring Trust on Your Teams (GOTO; Chicago 2020)
rmw
34
3.1k
Transcript
【 いまさら聞けない】 Swift の Optional ってなに? 2017/06/23 @fromatom
iOS 開発つまづきポイント Optional Delegate Autolayout ProvisioningPro fi le 突然ぶっ壊れる Xcode
突然ぶっ壊れる Xcode 突然ぶっ壊れる Xcode
iOS 開発つまづきポイント Optional Delegate Autolayout ProvisioningPro fi le 突然ぶっ壊れる Xcode
突然ぶっ壊れる Xcode 突然ぶっ壊れる Xcode
はじめに Swift の Optional が複雑( に見える) のは、 同じ記号(! と ?
) が違う意 味で使われているから たとえば、 下記で使われる ! と ? はすべて意味が違います var a: Int? = 1 var b: Int! = 2 let c: Int = a! + 10 self?.collectionView?.reloadData()
目次 1. Optional 2. Implicitly Unwrapped Optional 3. Optional Chaining
1. Optional
Swift での nil とは nil について知らないと Optional を理解できない Swift のnil
は値・ 式の評価結果がない( もしくはエラー) を表す Swift1.x 系では do-catch と throw が無かったんや Swift にはポインタがないので厳密には空ポインタではない
Swift での nil とは nil を通常の値と同じように触ると虚無に飲み込まれクラッシュする 自然界にnil はないので触ったら死ぬ 他の言語と違って0 やfalse
として解釈されない 虚無に型はない nil をハンドリングするために導入されたのがOptional
Optional Int Type まずは Int で考えます Int は整数型なので整数ではない nil は代入できない
nil を代入したい場合はInt? という Optional Int Type を使う var a: Int = 0 var b: Int? = 0 a = nil // コンパイルエラー になる b = nil // 代入できる
Optional Int Type var a: Int = 0 var b:
Int? = 0 var c: Optional<Int> = 0 // 上と同じ Int? はOptional<Int> のシンタックスシュガー Optional<T> というクラス( 正確には enum ) があり、Generics で様々 な型を入れられる つまり、 Int と Int? は別物 ← 超重要 型が違うので同じようには使えない ← 当たり前だよね
Optional Int Type を Int として使いたい Int? はInt ではないのでInt として扱えない
Unwrap することで使えるようになる ここで "!" が登場します Xcode に言われるがまま修正すると入ってくる謎の "!" "!" を使って Unwrap することを Force Unwrap と言います
Force Unwrap var optionalValue: Int? = 2000 var a: Int
= optionalValue + 20 // 型が違うのでコンパイルエラー になる var b: Int = optionalValue! + 20 // 2020 になる optionalValue! の "!" で Int を取り出している もしoptionalValue の中身がnil だったら実行時クラッシュ
条件判定 実行時クラッシュすると困るので条件判定をします var optionalValue: Int? = 2000 if optionalValue !=
nil { print(optionalValue!) // ここでは "!" が必要 } if optionalValue == 1000 { // "!" を書かなくても良い print("1000 だよ") } else if optionalValue == 2000 { print("2000 だよ") // これが表示される }
Optional Binding 毎回 "!" 書くのは大変 if 文の then 節では nil
チェックした変数を使いたいことが多い そこで Optional Binding です var optionalValue: Int? = 2000 if let value = optionalValue { // if-let で1 つの構文 print(value) //"!" が要らない } else { print("nil だったらしいよ") //optionalValue がnil なら実行される }
Optional Binding 複数の値を Optional Binding することもできます var optionalA: Int? =
1 var optionalB: Int? = 2 if let a = optionalA, let b = optionalB { // optionalA, optionalB が両方nil ではないときにくる print(a + b) // 3 が表示される }
条件式との組み合わせ optional binding した値を条件式で使える var optionalValue: Int? = 2000 if
let value = optionalValue, value > 100 { print("100 より大きいよ") }
Nil Coalescing Operator ある Optional Type の変数がnil だった時に、 他の定数を使いたいときに 便利なやつ
let value: Int = optionalValue ?? 100 optionalValue がnil なら100 が代入されます。 便利だけど多用すると右 に長くなっていくので注意。 let value: Int = optA ?? optB ?? optC ?? 100 ※Bool 値を見ているわけではないので注意
2. Implicitly Unwrapped Optional
Implicitly Unwrapped Optional 参照したら勝手に Unwrap してくれる Optional Type var a:
Int = 0 var b: Int! = 0 a = nil // コンパイルエラー になる b = nil // 代入できる Int! はImplicitlyUnwrappedOptional<Int> のシンタックスシュガー var b: Int! = 0 var c: ImplicitlyUnwrappedOptional<Int> = 0 // 上と同じ
Implicitly Unwrapped Optional "!" で Unwrap しなくてもよい var a: Int!
= 0 var b: Int = a + 20 // "!" マー クがなくてもアクセスできる! a = nil // a にnil をいれてみる var c: Int = a + 20 // 実行時にクラッシュする クラスの生成直後は nil だけど、1 度設定されたら nil になりえない変数に 使う。StoryBoard との関連(IBOutlet ) でよく使います。
3. Optional Chaining
Optional Chaining Method Chaining をする際、 途中で Optional な要素がある時に利用す る。 self?.collectionView?.reloadData()
self?.navigationController?.setNavigationBarHidden(false, animated: true)
Optional Chaining こういう class があり、3 世代前の名前を知りたい時 class Human { var
ancestor: Human? var name: String } こうやれば取れるけど、 どこかで nil が入ったらクラッシュ print(human.ancestor!.ancestor!.ancestor!.name)
Optional Chaining これでもいけるけど、 if let a = human.ancestor, let b
= a.ancestor, let c = b.ancestor { print(c.name) } Optional Chaining を使えばこう書ける if let name = human.ancestor?.ancestor?.ancestor?.name { print(name) }
メソッドも実行できる self?.collectionView?.reloadData() self?.navigationController?.setNavigationBarHidden(false, animated: true) 代入もできる // 3 世代前の先祖がいたら置き換える human.ancestor?.ancestor?.ancestor?
= newAncestor
ちょっと便利な Optional Chaining テク Void を返すメソッドを Optional Chaining で実行するとOptional<Void> を
返すようになる。 if self?.collectionView?.reloadData() != nil { //reloadData() が呼ばれたら実行される } if (human.ancestor?.ancestor?.ancestor? = newAncestor) != nil { // 代入が成功したら実行される }
まとめ これで Swift コー ド上で使われる様々 な意味の "!" と "?" の違いが(
ふわ っと) 分かったと思います。 var a: Int? = 1 // Optional var b: Int! = 2 // Implicitly Unwrapped Optional var c: Int = a! + 10 // Force Unwrap self?.collectionView?.reloadData() // Optional Chaining
参考文献 1. 詳解 Swift 改訂版 2016 年 1 月 1
日 初版第一刷発行 http://amzn.asia/eeKqCsH Swift3 系はこちら:http://amzn.asia/4IXtxem 2. どこよりも分かりやすい Swift の "?" と "!" http://qiita.com/maiki055/items/b24378a3707bd35a31a8