如何使用调用回调的 func 制作 swift 3.0 类?

JTh*_*ora 4 methods block swift

我需要简单的方法来创建一个类,我可以使用 Swift 3.0 传递和调用回调函数

我不喜欢在它们之前使用像通知中心或选择器这样的东西。我很挑剔,我也不喜欢打那么多字。

例子:

func setCallback(function:Callback){
    self.callback = function
}

func callCallback(){
    self.callback()
}
Run Code Online (Sandbox Code Playgroud)

Rob*_*Rob 6

假设您有一些复杂的任务,并且您想提供一个在任务完成时要调用的回调:

class ComplexTaskManager {
    var completionHandler: (() -> Void)?

    func performSomeComplexTask() {
        ...

        // when done, call the completion handler

        completionHandler?()

        // if this was a one-time completion handler, you'd `nil` it; it helps protect against strong reference cycles; otherwise comment the next line

        completionHandler = nil
    }
}
Run Code Online (Sandbox Code Playgroud)

所以,你会像这样使用它:

let manager = ComplexTaskManager()

manager.completionHandler = {
    // this is what I want to do when the task is done
}

manager.performSomeComplexTask()
Run Code Online (Sandbox Code Playgroud)

现在,我想象我们正在处理一些耗时的异步任务,您希望在完成后调用回调。但是还有其他类似的模式(例如,您可能正在遍历某个模型对象,并且想要为该对象中的每个实例调用回调),但其思想与上述相同。


如果您有一个强引用循环,您可以使用弱引用(例如供应[weak self][unowned self]闭包)来修复它。例如,想象:

class ViewController {
    let manager = ComplexTaskManager()

    @IBOutlet weak var statusLabel: UILabel!

    override viewDidLoad() {
        super.viewDidLoad()

        statusLabel.text = "Starting complex task"

        manager.completionHandler = { [weak self] in 
            self?.statusLabel.text = "Done"
        }

        manager.performSomeComplexTask()
    }
}
Run Code Online (Sandbox Code Playgroud)

但是只有在存在强引用循环时才需要这样做(例如,视图控制器保持对管理器的强引用,并且由于self闭包内部的存在,管理器保持对视图控制器的强引用)。但是,如果其中任何一个是本地引用而不是属性,则没有强引用循环可以打破。


坦率地说,对于每个任务调用一次的闭包,更常见的模式是将其作为方法的参数提供。例如:

class ComplexTaskManager {
    func performSomeComplexTask(completionHandler: () -> Void) {
        ...

        // when done, call the completion handler

        completionHandler()
    }
}
Run Code Online (Sandbox Code Playgroud)

所以,你会像这样使用它:

let manager = ComplexTaskManager()

manager.performSomeComplexTask() {
    // this is what I want to do when the task is done
}
Run Code Online (Sandbox Code Playgroud)

这种方法的美妙之处在于它不仅更简单,而且避免了强引用周期的风险。