gpt4 book ai didi

ios - 使用选择器错误的 Swift 3 协议(protocol)扩展

转载 作者:IT王子 更新时间:2023-10-29 05:11:49 25 4
gpt4 key购买 nike

我认为我的 UIViewController 有一个非常简单的协议(protocol)扩展,它提供了通过点击手势关闭键盘的功能。这是我的代码:

@objc protocol KeyboardDismissing { 
func on(tap: UITapGestureRecognizer)
}

extension KeyboardDismissing where Self: UIViewController {

func addDismissalGesture() {
let tap = UITapGestureRecognizer(target: self, action: #selector(Self.on(tap:)))
view.addGestureRecognizer(tap)
}

func on(tap: UITapGestureRecognizer) {
dismissKeyboard()
}

func dismissKeyboard() {
view.endEditing(true)
}
}

问题是上面的代码在这一行抛出编译错误:

let tap = UITapGestureRecognizer(target: self, action: #selector(Self.on(tap:)))

错误信息:

Argument of '#selector' refers to instance method 'on(tap:)' that is not exposed to Objective-C

建议通过在 func on(tap: UITapGestureRecognizer) 之前添加 @objc 来“修复它”

好的,我添加标签:

@objc func on(tap: UITapGestureRecognizer) {
dismissKeyboard()
}

但是,它会在这个新添加的 @objc 标记上抛出一个不同的编译错误,错误消息为:

@objc can only be used with members of classes, @objc protocols, and concrete extensions of classes

建议通过删除我刚刚被告知要添加的完全相同的标签来“修复它”

我最初认为在我的协议(protocol)定义之前添加 @objc 可以解决任何 #selector 问题,但显然情况并非如此,这些循环错误消息/建议不是丝毫帮助。我在各处添加/删除 @objc 标记、将方法标记为 optional、将方法放在协议(protocol)的定义中等等。

我在协议(protocol)定义中放入的内容也无关紧要 保持扩展不变,以下示例不起作用,协议(protocol)定义中声明的方法的任何组合也不起作用:

@objc protocol KeyboardDismissing { 
func on(tap: UITapGestureRecognizer)
}

这让我误以为它是通过编译为一个独立的协议(protocol)来工作的,但第二次我尝试将它添加到 View Controller 中:

class ViewController: UIViewController, KeyboardDismissing {}

它会吐出原来的错误。

谁能解释我做错了什么以及我该如何编译它?

注意:

我看过this question但它适用于 Swift 2.2 而不是 Swift 3,一旦您创建一个继承自示例中定义的协议(protocol)的 View Controller 类,答案也不会编译。

我也看过 this question但答案使用 NotificationCenter 这不是我想要的。

如果还有其他看似重复的问题,请告诉我。

最佳答案

马特的回答是正确的。但是,我只想补充一点,如果您正在处理 #selector 从 NotificationCenter 通知中使用,您可以尝试通过使用闭包版本来避免 #selector

例子:

而不是写:

extension KeyboardHandler where Self: UIViewController {

func startObservingKeyboardChanges() {

NotificationCenter.default.addObserver(
self,
selector: #selector(keyboardWillShow(_:)),
// !!!!!
// compile error: cannot be included in a Swift protocol
name: .UIKeyboardWillShow,
object: nil
)
}

func keyboardWillShow(_ notification: Notification) {
// do stuff
}
}

你可以这样写:

extension KeyboardHandler where Self: UIViewController {

func startObservingKeyboardChanges() {

// NotificationCenter observers
NotificationCenter.default.addObserver(forName: .UIKeyboardWillShow, object: nil, queue: nil) { [weak self] notification in
self?.keyboardWillShow(notification)
}
}

func keyboardWillShow(_ notification: Notification) {
// do stuff
}
}

关于ios - 使用选择器错误的 Swift 3 协议(protocol)扩展,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40296302/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com