Objective-Cでは、ブロックの入力と出力を定義し、メソッドに渡されたブロックの1つを保存して、後でそのブロックを使用できます。
// in .h
typedef void (^APLCalibrationProgressHandler)(float percentComplete);
typedef void (^APLCalibrationCompletionHandler)(NSInteger measuredPower, NSError *error);
// in .m
@property (strong) APLCalibrationProgressHandler progressHandler;
@property (strong) APLCalibrationCompletionHandler completionHandler;
- (id)initWithRegion:(CLBeaconRegion *)region completionHandler:(APLCalibrationCompletionHandler)handler
{
self = [super init];
if(self)
{
...
_completionHandler = [handler copy];
..
}
return self;
}
- (void)performCalibrationWithProgressHandler:(APLCalibrationProgressHandler)handler
{
...
self.progressHandler = [handler copy];
...
dispatch_async(dispatch_get_main_queue(), ^{
_completionHandler(0, error);
});
...
}
だから私はSwiftで同等のことをしようとしています:
var completionHandler:(Float)->Void={}
init() {
locationManager = CLLocationManager()
region = CLBeaconRegion()
timer = NSTimer()
}
convenience init(region: CLBeaconRegion, handler:((Float)->Void)) {
self.init()
locationManager.delegate = self
self.region = region
completionHandler = handler
rangedBeacons = NSMutableArray()
}
コンパイラーは、completionHandlerのその宣言を好みません。私がそれを責めているわけではありませんが、後でSwiftで設定して使用できるクロージャをどのように定義しますか?
コンパイラーは文句を言います
var completionHandler: (Float)->Void = {}
右側は適切な署名のクロージャーではないため、つまり、float引数を取るクロージャーです。次は、完了ハンドラーに「何もしない」クロージャーを割り当てます。
var completionHandler: (Float)->Void = {
(arg: Float) -> Void in
}
これは次のように短縮できます
var completionHandler: (Float)->Void = { arg in }
自動型推論による。
ただしおそらく必要なのは、Objective-Cインスタンス変数がnil
に初期化されるのと同じ方法で、完了ハンドラーがnil
に初期化されることです。 Swiftでは、これはoptionalで実現できます:
var completionHandler: ((Float)->Void)?
これで、プロパティはnil
(「値なし」)に自動的に初期化されます。 Swiftでは、オプションのバインディングを使用して、完了ハンドラーに値があることを確認します
if let handler = completionHandler {
handler(result)
}
またはオプションの連鎖:
completionHandler?(result)
Objective-C
@interface PopupView : UIView
@property (nonatomic, copy) void (^onHideComplete)();
@end
@interface PopupView ()
...
- (IBAction)hideButtonDidTouch:(id sender) {
// Do something
...
// Callback
if (onHideComplete) onHideComplete ();
}
@end
PopupView * popupView = [[PopupView alloc] init]
popupView.onHideComplete = ^() {
...
}
スイフト
class PopupView: UIView {
var onHideComplete: (() -> Void)?
@IBAction func hideButtonDidTouch(sender: AnyObject) {
// Do something
....
// Callback
if let callback = self.onHideComplete {
callback ()
}
}
}
var popupView = PopupView ()
popupView.onHideComplete = {
() -> Void in
...
}
これがあなたが望んでいるものかどうかわからない例を提供しました。
var completionHandler: (value: Float) -> ();
func printFloat(value: Float) {
println(value)
}
completionHandler = printFloat
completionHandler(value: 5)
宣言されたcompletionHandler
変数を使用して、単に5を出力します。
Swift 4.に、2つのパラメータディクショナリとブールを含むクロージャ変数を作成しました。
var completionHandler:([String:Any], Bool)->Void = { dict, success in
if sucess {
print(dict)
}
}
クロージャー変数を呼び出す
self.completionHandler(["name":"Gurjinder singh"],true)
クロージャーは、以下のようにtypealias
として宣言できます。
typealias Completion = (Bool, Any, Error) -> Void
コード内の任意の場所で関数で使用する場合。あなたは通常の変数のように書くことができます
func xyz(with param1: String, completion: Completion) {
}
これも機能します:
var exeBlk = {
() -> Void in
}
exeBlk = {
//do something
}
//instead of nil:
exeBlk = {}