繁体   English   中英

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

[英]Generic protocol method swift can't recognise type as type

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

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

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

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

class MyClass {
    let a = "sometext"
}

在该类中,我想从中获取所选的项,我实现了该方法,并为所选的项委派了一个用于列表中对象类型的类型参数。

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
}

当然有更好的方法吗? 还是这是敏捷的缺点?

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

let selected = selectedItem as! MyClass

它只是强制铸造成自己的类型,几乎可以是任何东西。 MyClass在委托方法的主体采用的是在函数签名使用的通用参数,而不是MyClass具有类a构件。

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

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

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

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

    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类型的选定项一起使用的委托实现可能如下所示:

    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)")
        }
    }

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM