swift - 通用协议(protocol)方法 swift 无法将类型识别为类型

标签 swift generics swift3 polymorphism swift-protocols

我已经创建了一个通用 View Controller 来从列表中选择一个项目,我想使用一个通用委托(delegate)方法来返回已选择的项目。我的设置如下:

class ListPickerViewController<T>: UIViewController {
    var delegate: ListPickerViewControllerDelegate?
}

protocol ListPickerViewControllerDelegate {
    func listPickerViewControllerDelegate<T>(_ listPickerViewController: ListPickerViewController<T>, selectedItem: T)
}

我有一个对象列表,我将使用这些对象来填充 ListPickerViewController

class MyClass {
    let a = "sometext"
}

在我想从中获取所选项目的类中,我实现了该方法并为所选项目委托(delegate)提供了列表中对象类型的类型参数。

func listPickerViewControllerDelegate<MyClass>(_ listPickerViewController: ListPickerViewController<MyClass>, selectedItem: MyClass) {
    print(selectedItem.a)
}

然而,这并不能编译!我收到错误“MyClass 类型的值没有成员 a”。所以我尝试将其转换为类型

func listPickerViewControllerDelegate<MyClass>(_ listPickerViewController: ListPickerViewController<MyClass>, selectedItem: MyClass) {
    let selected = selectedItem as! MyClass
}

这给出了“将 MyClass 强制转换为相同类型无效”。

出于某种原因,我发现这行得通:

typealias MyClassAlias = MyClass 

func listPickerViewControllerDelegate<MyClass>(_ listPickerViewController: ListPickerViewController<MyClass>, selectedItem: MyClass) {
    let selected = selectedItem as MyClassAlias
    print(selected.a) // works
}

肯定有更好的方法吗?或者这是swift的缺点?

最佳答案

在你的协议(protocol)实现中委托(delegate)方法的签名中 MyClass 只是一个通用参数,你可以使用 T, K,等。它不一定有一个名为 a 的成员,因此是第一个编译错误。在

let selected = selectedItem as! MyClass

它只是强制转换为它自己的类型,几乎可以是任何类型。委托(delegate)方法主体中使用的 MyClass 是函数签名中使用的通用参数,而不是具有 a 成员的 MyClass 类。

如果您使用类型别名,则可以在方法主体中使用它并进行编译,但是一旦 MyClass 泛型参数的实际类型更改为 MyClass 类,转换将在运行时失败。

通常,您可能需要为每种类型的选定对象实现单独的委托(delegate)协议(protocol)实现。我假设 View Controller 的通用参数 T 是选定的对象类型。您还需要确保特定类型的 View Controller 使用正确的委托(delegate)实现。

在 Swift 中,您可以使用关联类型来完成此操作。参见 https://developer.apple.com/library/content/documentation/Swift/Conceptual/Swift_Programming_Language/Generics.html .

View Controller 和委托(delegate)协议(protocol)可能如下所示:

    class ListPickerViewController<T, D : ListPickerViewControllerDelegate>
    : UIViewController
    where D.T == T
    {
        var delegate: D?
    ......
    }

    protocol ListPickerViewControllerDelegate {
        associatedtype T
        func listPickerViewControllerDelegate<D>(_ controller: 
ListPickerViewController<T, D>, selectedItem: T) where D.T == T
    }

MyClass 类型的选定项一起使用的委托(delegate)实现可能如下所示:

    class MyDelegateImpl : ListPickerViewControllerDelegate {
        typealias T = MyClass

        internal func listPickerViewControllerDelegate<D>(_ controller: 
ListPickerViewController<MyClass, D>, selectedItem: MyClass) 
    where D.T == MyClass {
            print("In the delegate method!")
            print("MyClass.a = \(selectedItem.a)")
        }
    }

关于swift - 通用协议(protocol)方法 swift 无法将类型识别为类型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42668850/

相关文章:

ios - 如何处理与 NSURLSession 失去互联网连接?

c# - 防止基于属性值更改对象的实例

java - 在 Java 中添加两个泛型(强制转换不起作用)

ios - 使用 swift 3 更改 iOS 中的按钮标题不起作用

ios - 为什么在延迟调用时循环迭代有时会被打乱? [代码和输出]

ios - 使用 UIScrollView 创建封面流

ios - 如何在推送其 View Controller 之前预加载 WKWebView?

actionscript-3 - 可以使用类型引用声明 AS3 向量吗?

ios - 方法没有覆盖父类(super class)中的任何函数(覆盖 func collectionView)

swift - 可选变量内的变量