我无法弄清楚如何进行演员表演让我最终能够引入一些与泛型一起工作的动态. 下一段代码无法编译.它显示了这个错误: Cannot invoke ‘createContainer’ with an argument list of type ‘(FooProtoco
下一段代码无法编译.它显示了这个错误:
Cannot invoke ‘createContainer’ with an argument list of type
‘(FooProtocol)’ Expected an argument list of type ‘(T)’
protocol FooProtocol { func doSomething() } class Foo : FooProtocol { func doSomething() {} } class Container<T : FooProtocol> { let someDataConformingFooProtocol : T init(someDataConformingFooProtocol : T) { self.someDataConformingFooProtocol = someDataConformingFooProtocol } } class AllTogether { init () { createContainer(Foo()) //So far, so good let foo2Mask : AnyObject = Foo() if let foo2MaskChecked = foo2Mask as? FooProtocol { createContainer(foo2MaskChecked) //ERROR: Cannot invoke 'createContainer' with an argument list of type '(FooProtocol)' //Expected an argument list of type '(T)' } } func createContainer<T : FooProtocol>(data: T){ Container<T>(someDataConformingFooProtocol: data) } }
这真的是预期的行为吗?因为我个人无法理解编译器抱怨它的原因或原因.
什么是合适的演员?没有引用具体类,我的意思是不喜欢这样:
if let foo2MaskChecked = foo2Mask as? Foo
谢谢!
它归结为T:FooProtocol和FooProtocol之间的区别,正如@Hamish在评论中提到的那样.当我有这样的功能:
func foo(_ i: FooProtocol)
我正在接受类型为FooProtocol的实例,因此我可以传入一个符合FooProtocol的值,或者它的类型是FooProtocol.这类似于子类化,您可以将SuperClass和SubClass传递给SuperClass.
但是如果你有这样的功能:
func foo<T>(_ i: T) where T: FooProtocol
我可以传入任何符合FooProtocol的类型,但由于FootProtocol不符合自身,你无法传递它.
所以在你的问题中,你试图传入一个类型FooProtocol,其中的类型必须符合FooProtocol.您可以通过以下方式解决此问题:
class Container { let someDataConformingFooProtocol: FooProtocol init(someDataConformingFooProtocol: FooProtocol) { self.someDataConformingFooProtocol = someDataConformingFooProtocol } } // Later func createContainer(data: FooProtocol){ Container(someDataConformingFooProtocol: data) }