通用协议方法swift can';别把类型认作类型

通用协议方法swift can';别把类型认作类型,swift,generics,swift3,polymorphism,swift-protocols,Swift,Generics,Swift3,Polymorphism,Swift Protocols,我已经创建了一个通用视图控制器来从列表中选择一个项目,我想使用一个通用委托方法来返回已选择的项目。我的设置如下: class ListPickerViewController<T>: UIViewController { var delegate: ListPickerViewControllerDelegate? } protocol ListPickerViewControllerDelegate { func listPickerViewControllerD

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

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

protocol ListPickerViewControllerDelegate {
    func listPickerViewControllerDelegate<T>(_ listPickerViewController: ListPickerViewController<T>, selectedItem: T)
}
    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)")
        }
    }
在我想要从中获取所选项的类中,我实现了该方法,并为所选项委托提供了列表中对象类型的类型参数

func listPickerViewControllerDelegate<MyClass>(_ listPickerViewController: ListPickerViewController<MyClass>, selectedItem: MyClass) {
    print(selectedItem.a)
}
func listPickerViewControllerDelegate(\listPickerViewController:listPickerViewController,selectedItem:MyClass){
打印(选择editem.a)
}
然而,这并不编译!我收到一个错误“MyClass类型的值没有成员a”。所以我试着把它铸造成那种类型

func listPickerViewControllerDelegate<MyClass>(_ listPickerViewController: ListPickerViewController<MyClass>, selectedItem: MyClass) {
    let selected = selectedItem as! MyClass
}
func listPickerViewControllerDelegate(\listPickerViewController:listPickerViewController,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
}
typealias MyClassAlias=MyClass
func listPickerViewController delegate(\listPickerViewController:listPickerViewController,selectedItem:MyClass){
let selected=selectedItem作为MyClassAlias
打印(selected.a)//工作
}

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

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

let selected = selectedItem as! MyClass
它只是强制转换为自己的类型,几乎可以是任何类型<委托方法主体中使用的code>MyClass是函数签名中使用的泛型参数,而不是具有
a
成员的
MyClass

如果使用typealias,则可以在方法体中使用它并进行编译,但只要
MyClass
泛型参数的实际类型更改为
MyClass
类以外的类型,转换就会在运行时失败

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

在Swift中,您可以使用关联类型来完成此操作。看

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

    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
只是一个通用参数,您可以使用
T
K
等。它不一定有名为
a
的成员,因此出现了第一个编译错误。在

let selected = selectedItem as! MyClass
它只是强制转换为自己的类型,几乎可以是任何类型<委托方法主体中使用的code>MyClass是函数签名中使用的泛型参数,而不是具有
a
成员的
MyClass

如果使用typealias,则可以在方法体中使用它并进行编译,但只要
MyClass
泛型参数的实际类型更改为
MyClass
类以外的类型,转换就会在运行时失败

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

在Swift中,您可以使用关联类型来完成此操作。看

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

    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
    }

抱歉-我的意思是删除WBSCode并替换为MyClass,现在更新了MyClass为什么引用MyClass而不是实际的类的解释。那么我们如何才能告诉swift这种方法应该是什么类型的呢?或者-它总是一个泛型类型,这取决于你来铸造它吗?对不起-我的意思是删除WBSCode并替换为MyClass,现在更新了MyClass为什么引用而不是实际类的解释。那么我们如何才能告诉swift这种方法应该是什么类型的呢?或者-它总是一个泛型类型,由你来决定它的类型?