詳解Swift改訂版(13)〜クロージャ

2017. 03. 30
この記事は詳解 Swift 改訂版(初版第1刷)を元に、Xcode 8.2.1(Swift 3.0.2)下における差異を記述しています。



CHAPTER 13 クロージャ



●13.1 クロージャの宣言


・クロージャと関数の型

単一引数の関数を定数に代入する例において、引数型を括る () を省略するとエラーが出るようになりました。

let c4: Int -> Double = f2    // error: single argument function types require parentheses

詳解Swift 第3版」の同項によると、Swift 3 でこの記法は廃止されたようです。


・クロージャの複雑な型宣言

クロージャを格納する空配列を変数に代入する例で、本文ではエラーになるとありますが、現在は問題ないようです。

var ca3 = [(Int, Int) -> Double]()
type(of: ca3)    // Array<((Int, Int)) -> Double>.Type



●13.2 変数のキャプチャ


・クロージャが参照型の変数をキャプチャする場合

List13-3 において、「詳解Swift改訂版(03)〜構造体」の「3.3 プロパティ/格納型プロパティの初期値を式で設定する」で述べたように Swift 3 でインクリメント演算子は廃止されており、且つ文字列埋め込みに複合代入演算子を用いた式は使えませんので、関数 makerZ の return 文は以下のように修正することになります。

print("\(s): \(++localvar.value)")    // error: '++' is unavailable: it has been removed in Swift 3

print("\(s): \(localvar.value += 1)")    // エラーや警告は出ないが「m1: ()」など式が反映されない

localvar.value += 1
print("\(s): \(localvar.value)")


・イニシャライザとクロージャ

列挙型 Lot の例において、クロージャ gen への引数としてタプル値を渡すとエラーが出るようになりました。
(「Developers.IO/[iOS][Swift] Swift 3.0の変更点まとめ」の "SE-0029" 参照)

Swift 3 から複数の引数に対して単一のタプル値を渡すことはできなくなったようですが、タプル値を展開して各々の引数に渡すのは問題ありません。

lots.append( gen(t) )    // error: passing 2 arguments to a callee as a single tuple value has been removed in Swift 3

lots.append( gen(t.0, t.1) )



●13.3 クロージャの使い方と記法


・配列の整列

配列の要素を並び替える例において、元配列 list にメソッド sort を指定するとエラーが出るようになりました。
(「Developers.IO/[iOS][Swift] Swift 3.0の変更点まとめ」の "SE-0006" 参照)

Swift 3 からメソッド sort は sorted(by:) に改名されたので修正する必要があります。

let slist = list.sort( { (a:String, b:String) -> Bool in a < b } )    // error: 'sort' has been renamed to 'sorted(by:)'

let slist = list.sorted(by: { (a:String, b:String) -> Bool in a < b } )


・配列要素の選択

List13-6 において、関数 select の引数定義でエラーが出るようになりました。

Swift 3 から関数の型を定義する際に引数を囲む () は省略できなくなったので追加する必要があります。

func select(_ list:[String], _ filter:String->Bool) -> ([String],[String]) {    // error: single argument function types require parentheses

func select(_ list:[String], _ filter:(String)->Bool) -> ([String],[String]) {


・エラーを投げるクロージャ

定数 qux に代入しているクロージャの中で唐突に出てくる XError.Zero はエラーの種類を示す列挙型で、「詳解 Swift 改訂版 サンプルダウンロード」の「samples/Chap13/p316.swift」からコピーすると便利です。

ただし「詳解Swift改訂版(12)〜エラー処理」の「12.1 エラー処理構文/エラーの発生」で述べたように、プロトコル ErrorType は Error に改名されたとエラーが出るので修正する必要があります。

enum XError : ErrorType {    // error: 'ErrorType' has been renamed to 'Error'

enum XError : Error {

また本文中で「Int throws -> Double」と書いても同じとありますが、上記の通り引数を囲む () を省略することはできません。

var syr : (Int) throws -> Double = qux

var syr : Int throws -> Double = qux    // error: single argument function types require parentheses


・rethrow のある関数

List13-7 において、引数の記述でいくつか修正する必要があります。

まず「詳解Swift改訂版(02)〜関数」の「2.1 関数定義の基本/外部引数名」で述べたように、関数の第1引数は自動的に外部引数名となるので呼び出し時に省略したい場合は「_」を追加します。

また inout 引数は「詳解Swift改訂版(02)〜関数」の「2.2 関数定義におけるさまざまな設定/inout 引数」で述べたように、: の後ろに書く、さらに第2引数のクロージャでは上述の通り引数を囲む () を追加します。

func apply(inout arg:[Int], _ map:Int throws -> Int) rethrows {
// 'inout' before a parameter name is not allowed. place it before the parameter type instead
// error: single argument function types require parentheses


func apply(_ arg:inout [Int], _ map:(Int) throws -> Int) rethrows {

XError 型は「詳解 Swift 改訂版 サンプルダウンロード」の「samples/Chap13/List13-7.swift」で定義されています。
(上述の p316.swift のものと同じです)


・配列要素に対する操作

2つ目の配列 dat から複数の条件で新たな文字列配列 str を得る例において、上述の通りメソッド sort は sorted(by:) に改名されたので修正する必要があります。

let str_ = dat.filter{ $0 >= 10 }.sort( < ).map{ "\($0)" }    // error: 'sort' has been renamed to 'sorted(by:)'

let str_ = dat.filter{ $0 >= 10 }.sorted(by: < ).map{ "\($0)" }

またSequeceType プロトコルの reduce メソッドも reduce(_:_:) に改名されたので修正する必要があります。

let sum = numbers.reduce(0, combine:+)    // error: 'reduce(_:combine:)' has been renamed to 'reduce(_:_:)'

let sum = numbers.reduce(0, +)


・関数のカリー化と部分適用

詳解Swift(12)〜クロージャ」の「12.3 クロージャの使い方と記法/関数のカリー化と部分適用」で述べたように、Swift 3 でカリー化関数構文は廃止されました。
(「Developers.IO/[iOS][Swift] Swift 3.0の変更点まとめ」の SE-0002 参照)



●13.4 クロージャと強い参照の循環


・キャプチャによる強い参照

List13-8 において、遅延格納プロパティ daytime と nighttime の定義で引数を囲む () と関数 meter() を呼び出す際の引数ラベル midnight: を追加する必要があります。

lazy var daytime: Int->Int = self.meter(false)    // error: single argument function types require parentheses
// error: missing argument label 'midnight:' in call

lazy var nighttime: Int->Int = self.meter(true)    // error: single argument function types require parentheses
// error: missing argument label 'midnight:' in call


lazy var daytime: (Int)->Int = self.meter(midnight: false)
lazy var nighttime: (Int)->Int = self.meter(midnight: true)

また関数 meter() の定義においても、引数を囲む () を追加する必要があります。

func meter(midnight: Bool) -> (Int -> Int) {    // error: single argument function types require parentheses

func meter(midnight: Bool) -> (Int) -> Int {

それと関数 meter() 内のクロージャ式で、「詳解Swift改訂版(02)〜関数」の「2.2 関数定義におけるさまざまな設定/引数の値を処理中に変更できるようにする」で述べたように、引数 distance に変数 var を指定することはできなくなりましたので、(定数の)引数 dist で受けて 変数 distance に入れ替えるなどの修正をする必要があります。

return { (var distance: Int) in    // error: parameters may not have the 'var' specifier

return { (dist:Int) in
            var distance = dist

実行例においては、C 様式の for 文が使われているので for-in 文に修正します。

for var d = 1000; d <= 5000; d += 1000 {    // error: C-style for statement has been removed in Swift 3

for d in stride(from: 1000, through: 5000, by: 1000) {


・関数の引数にクロージャを渡す場合

最初の例の関数 setFunc() の定義で、離脱する(@escaping)クロージャに(暗黙的に)非離脱パラメータである f を割り当てようとしているとエラーが出るようになりました。

func setFunc(_ f:(Int) -> Int) { theFunc = f }
// error: assigning non-escaping parameter 'f' to an @escaping closure
// note: parameter 'f' is implicitly non-escaping

Qiita/Xcode 8 Release Notes 日本語翻訳メモ(SE-0103)」によると、Swift 3 でクロージャの引数は @noescape の動作がデフォルトとなり、離脱する場合は @escaping で明示することになったようです。

これに関しては「Xcode Release Notes(Xcode 8.0/Swift/New Features)」に、その旨の記述があります。

「クロージャ・パラメータは、@noescape で明示的に注釈をつけるのではなく、デフォルトで非離脱となっています。
クロージャ・パラメータが離脱する場合は @escaping を使用して示します。
@autoclosure(escaping) は現在 @autoclosure @escaping と記述します。
注釈 @noescape と @autoclosure(escaping) は廃止されました。
(
SE-0103)」

また「Developers.IO/[iOS][Swift] Swift 3.0の変更点まとめ(SE-0049)」によると、Swift 3 から @noescape の記述位置がパラメータの属性からパラメータ型の属性に変わったとあり、@escaping も : の後ろに記述することになります。

したがって関数 setFunc() は以下のようになります。

func setFunc(_ f:@escaping (Int) -> Int) { theFunc = f }



Developers.IO/[iOS][Swift] Swift 3.0の変更点まとめ

Qiita/Xcode 8 Release Notes 日本語翻訳メモ

詳解Swift 第3版(Amazon)
 






QuietControl 30 wireless headphones
0 Comments
Leave a comment
管理者にだけ表示を許可する
Top
0 Trackbacks
Top
Calendar
04 | 2017/05 | 06
Sun Mon Tue Wed Thu Fri Sat
- 1 2 3 4 5 6
7 8 9 10 11 12 13
14 15 16 17 18 19 20
21 22 23 24 25 26 27
28 29 30 31 - - -
Recent Articles
iTunes


Swift
Categories
Tips
Profile

水月杏香

Author:水月杏香
永遠の初心者プログラマ。

Wish List
WACOM


ARC
Technical Q&A
情報プロパティリストキー
Start Developing iOS Apps Today
BOSE

QuietControl 30 wireless headphones
Reference
NSApplicationDelegateプロトコル
NSArrayクラス
NSAutoreleasePoolクラス
NSBundleクラス
NSBundle UIKit追加分
NSCalendarクラス
NSCoderクラス
NSCodingプロトコル
NSCopyingプロトコル
NSDataクラス
NSDateクラス
NSDateFormatterクラス
NSDictionaryクラス
NSEntityDescriptionクラス
NSEnumeratorクラス
NSErrorクラス
NSExceptionクラス
NSFetchRequestクラス
NSFileHandleクラス
NSFileManagerクラス
NSIndexPathクラス
NSIndexPath UIKit追加分
NSKeyedArchiverクラス
NSKeyedUnarchiverクラス
NSKeyValueCodingプロトコル
NSLocaleクラス
NSManagedObjectクラス
NSManagedObjectContextクラス
NSManagedObjectModelクラス
NSMutableArrayクラス
NSMutableCopyingプロトコル
NSMutableDictionaryクラス
NSMutableSetクラス
NSNotificationクラス
NSNotificationCenterクラス
NSNullクラス
NSNumberクラス
NSObjectクラス
NSObject UIKit追加分
NSObjectプロトコル
NSPersistentStoreクラス
NSPersistentStoreCoordinatorクラス
NSPredicateクラス
NSPropertyListSerializationクラス
NSRunLoopクラス
NSSetクラス
NSStringクラス
NSString UIKit追加分
NSTimerクラス
NSTimeZoneクラス
NSURLクラス
NSURLProtectionSpaceクラス
NSURLRequestクラス
NSUserDefaultsクラス
NSValueクラス

UIActionSheetクラス
UIActionSheetDelegateプロトコル
UIActivityIndicatorViewクラス
UIAlertViewクラス
UIAlertViewDelegateプロトコル
UIApplicationクラス
UIApplicationDelegateプロトコル
UIBarButtonItemクラス
UIBarItemクラス
UIButtonクラス
UIColorクラス
UIControlクラス
UIDatePickerクラス
UIDeviceクラス
UIEventクラス
UIFontクラス
UIGestureRecognizerクラス
UIImageクラス
UIImageViewクラス
UIKit Function
UILabelクラス
UINavigationControllerクラス
UINavigationItemクラス
UIPickerViewクラス
UIPickerViewDataSourceプロトコル
UIPickerViewDelegateプロトコル
UIPinchGestureRecognizerクラス
UIResponderクラス
UIScreenクラス
UIScrollViewクラス
UISearchBarクラス
UISearchBarDelegateプロトコル
UISegmentedControlクラス
UISliderクラス
UISwipeGestureRecognizerクラス
UISwitchクラス
UITableViewクラス
UITableViewCellクラス
UITableViewControllerクラス
UITableViewDataSourceプロトコル
UITableViewDelegateプロトコル
UITapGestureRecognizerクラス
UITextFieldクラス
UITextFieldDelegateプロトコル
UITextInputTraitsプロトコル
UITextViewクラス
UITextViewDelegateプロトコル
UIToolbarクラス
UITouchクラス
UIViewクラス
UIViewControllerクラス
UIWebViewクラス
UIWebViewDelegateプロトコル
UIWindowクラス

AVAudioPlayerクラス
AVAudioPlayerDelegateプロトコル

CADisplayLinkクラス
CAEAGLLayerクラス
CALayerクラス

CGAffineTransform
CGBitmapContext
CGColor
CGColorSpace
CGContext
CGGeometry
CGImage
CGPath

EAGLContextクラス
EAGLDrawableプロトコル

Foundation Constants
Foundation Data Types
Foundation Functions

MPMediaItemクラス
MPMediaItemArtworkクラス
MPMediaPlaylistクラス
MPMediaPropertyPredicateクラス
MPMediaQueryクラス
MPMusicPlayerControllerクラス

Randomization Services

System Sound Services
Amazon


OpenGL ES
SQLite
Monthly Archives
Recent Comments
Recent TrackBacks
RSS Link
Visitors
QR Code
QR