詳解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)
 

詳解Swift改訂版(12)〜エラー処理

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



CHAPTER 12 エラー処理



●12.1 エラー処理構文


・エラーの発生

TicketError 型の定義において、プロトコル ErrorType は Error に改名されたとエラーが出るので修正する必要があります。

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

enum TicketError : Error {


・エラーのあるオプショナル型の関数と try? 演算子

List12-3 において、if-let 文で try? 演算子による関数 convInt の判定をした場合、定数 x の標準出力で警告が出ます。

print("x:", x)    // Expression implicitly coerced from 'Int?' to Any

try 演算子であれば定数 x は整数に変換できた場合は Int 型、できなかった場合は nil となりelse 節が実行、空文字列の場合は(上流もなく捕捉されずに)何もしないことになります。

しかし try? 演算子ですと(If-let 文でも)定数 x はオプショナル Int 型になり、整数への変換の成否に関わらず x の print 文が実行されるために警告が出るので、nil 合体演算子で修正すると良いでしょう。

print("x:", x ?? "nil")

空文字列の時はエラーが発生すると(上流に遡って捕捉しようとせず) nil を返すため、else 節が実行され Failure が出力されます。

if-case 文での定数 x は、整数に変換できた場合は Int 型として x の print 文が実行され、できなかった場合はオプショナル nil に、エラーが発生した場合は nil になり else 節が実行されます。



●12.3 アクセス制御


・プロパティのアクセス制御

List12-8 において、サブクラス WebCamera の定義でエラーが出るようになりました。

class WebCamera : USBCamera {
    internal override var focus: Double {    // error: property does not override any property from its superclass
        didSet {
            super.shutterSpeed = ss(super.focus)    // error: 'shutterSpeed' is inaccessible due to 'private' protection level
        }
    }
    internal private(set) override var shutterSpeed: Double {    // error: property does not override any property from its superclass
        get { return super.shutterSpeed }    // error: 'shutterSpeed' is inaccessible due to 'private' protection level
        set { super.shutterSpeed = newValue }    // error: 'shutterSpeed' is inaccessible due to 'private' protection level
    }
    override init() { super.init() }
    override func take() { super.take() }    // error: method does not override any method from its superclass
    private func ss(_ f: Double) -> Double {
        return ( f * f) / 4000.0
    }
}

プロパティやメソッドがオーバーライドできない、または private なのでアクセスできないというものですが、「Qiita/Xcode 8 Release Notes 日本語翻訳メモ」によると、Swift 3 から元々の「private」は「fileprivate」になったようです。

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

「private として記された宣言は、宣言されている(基本的に波括弧 {} で囲まれている)レキシカル・スコープ内でしかアクセスすることはできません。
Swift 2 では、ファイルのトップレベルでの private 宣言はそのファイルのどこにでもアクセスすることができました。
以前は private として知られていたアクセスレベルは、現在は fileprivate と呼ばれています。
(
SE-0025)」

つまり定義されたソースファイル内でのみアクセス可能にするには private ではなく fileprivate に、private はクラスや構造体の定義内でのみアクセスする場合に用いると変更されました。

したがってクラス USBCamera およびサブクラス WebCamera の private を fileprivate に変更する必要があります。

class USBCamera {
    fileprivate var focus = 2.0
    fileprivate var shutterSpeed = 1.0 / 1000.0
    fileprivate func take() {
        print("take a picture")
    }
}

class WebCamera : USBCamera {
    internal override var focus: Double {
        didSet {
            super.shutterSpeed = ss(super.focus)
        }
    }
    internal fileprivate(set) override var shutterSpeed: Double {
        get { return super.shutterSpeed }
        set { super.shutterSpeed = newValue }
    }
    override init() { super.init() }
    override func take() { super.take() }
    fileprivate func ss(_ f: Double) -> Double {
        return ( f * f) / 4000.0
    }
}

なお「詳解Swift(8)〜クラスと継承」の「8.5 アクセス制御/プロパティのアクセス制御」で述べたように、プレイグラウンドではこの可視性の実行例を試すことができません。



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

詳解Swift 第3版(Amazon)
 






Bose SoundLink around-ear wireless headphones II
Calendar
02 | 2017/03 | 04
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

Bose SoundLink around-ear wireless headphones II
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