通用协议方法swift无法将类型识别为类型

时间:2017-03-08 10:31:14

标签: swift generics swift3 polymorphism swift-protocols

我创建了一个通用视图控制器来从列表中选择一个项目,我想使用泛型委托方法来返回已选择的项目。我的设置如下:

BEGIN
DECLARE @val INT;

WHILE 'toto' = 'titi'
BEGIN
    DECLARE @val int
    SELECT * FROM randomtable1
    SELECT * FROM randomtable2  
END
END 

我有一个对象列表,我将用它来填充ListPickerViewController

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

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

在类中,我希望从中获取所选项目,我实现了该方法,并为所选项目委托了列表中对象类型的类型参数。

class MyClass {
    let a = "sometext"
}

然而这不编译!我收到错误'MyClass类型的值没有成员'。所以我尝试将它转换为

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

这使'MyClass强制转换为相同类型无效'。

出于某种原因,我发现这有效:

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

当然有更好的方法吗?或者这是一个快速的缺点?

1 个答案:

答案 0 :(得分:2)

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

let selected = selectedItem as! MyClass

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

如果使用typealias,那么它可以在方法体中使用并编译,但只要MyClass泛型参数的实际类型更改为{{1}以外的其他类型, } class,强制转换将在运行时失败。

通常,您可能需要为每种类型的所选对象单独的委托协议实现。我假设视图控制器的通用参数MyClass是选定的对象类型。您还需要确保特定于类型的视图控制器使用正确的委托实现。

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

视图控制器和委托协议可能如下所示:

T

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