在Objective-C中,您可以定义块的输入和输出,存储其中一个传入方法的块,然后再使用该块:
// 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中设置和使用的闭包?
答案 0 :(得分:299)
编译器抱怨
var completionHandler: (Float)->Void = {}
因为右侧不是适当签名的封闭,即封闭 一个浮动参数。以下将为此分配一个“无所事事”的闭包 完成处理程序:
var completionHandler: (Float)->Void = {
(arg: Float) -> Void in
}
这可以缩短为
var completionHandler: (Float)->Void = { arg in }
由于自动类型推断。
但是你可能想要的是将完成处理程序初始化为nil
与将Objective-C实例变量初始化为nil
的方式相同。在斯威夫特
这可以通过可选:
var completionHandler: ((Float)->Void)?
现在该属性自动初始化为nil
(“无值”)。
在Swift中,您将使用可选绑定来检查
完成处理程序具有值
if let handler = completionHandler {
handler(result)
}
或可选链接:
completionHandler?(result)
答案 1 :(得分:33)
<强>目标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
...
}
答案 2 :(得分:5)
我提供了一个例子,不确定这是否是你所追求的。
var completionHandler: (value: Float) -> ();
func printFloat(value: Float) {
println(value)
}
completionHandler = printFloat
completionHandler(value: 5)
它只使用声明的completionHandler
变量打印5。
答案 3 :(得分:5)
在swift 4.我创建了一个包含两个参数字典和bool的闭包变量。
var completionHandler:([String:Any], Bool)->Void = { dict, success in
if sucess {
print(dict)
}
}
调用闭包变量
self.completionHandler(["name":"Gurjinder singh"],true)
答案 4 :(得分:3)
闭包可以声明为typealias
,如下所示
typealias Completion = (Bool, Any, Error) -> Void
如果你想在你的函数中使用代码中的任何地方;你可以写成普通变量
func xyz(with param1: String, completion: Completion) {
}
答案 5 :(得分:2)
这也有效:
var exeBlk = {
() -> Void in
}
exeBlk = {
//do something
}
//instead of nil:
exeBlk = {}
答案 6 :(得分:1)
根据您的需要,除了已接受的答案外,还有一个。你也可以这样实现:
var parseCompletion: (() ->Void)!
然后在一些函数中分配给它
func someHavyFunc(completion: @escaping () -> Void){
self.parseCompletion = completion
}
并在第二个函数中使用它
func someSecondFunc(){
if let completion = self.parseCompletion {
completion()
}
}
注意 @escaping
参数在这里是必填的
答案 7 :(得分:-1)
对我来说,以下是工作:
var completionHandler:((Float)->Void)!