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



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

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
}

肯定有更好的方法?还是这是Swift的缺点?

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

let selected = selectedItem as! MyClass

它只是其自己类型的力量,这几乎可以是任何东西。MyClass在委托方法的正文中使用的是功能签名中使用的通用参数,而不是具有a成员的MyClass类。

如果您使用typealias,则可以在方法的正文中使用它,并在编译中使用,但是一旦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)")
        }
    }

最新更新