使Swift类符合协议 – 在静态/类级别

我试图在Swift中构build一个通用的UITableViewController子类,它将容纳任何数量的不同types的表视图单元,而不需要内部知识。

要做到这一点,我试图使用我的模型和我的表格视图单元格的协议。 模型的协议将返回我应该去的单元类,单元的协议将返回问题的答案,例如单元格的高度应该是给定模型的答案。

但是我在协议的工作上遇到了问题,因为在第二个协议中,我想要转到单元的类而不是实例。

该模型的协议如下:

protocol JBSTableItemDelegate { func tableCellDelegate() -> JBSTableViewCellInterface } 

单元的协议如下:

 protocol JBSTableViewCellInterface: class { static func registerNibsWithTableView(tableView: UITableView) static func tableView(tableView: UITableView, heightForRowAtIndexPath indexPath: NSIndexPath?, tableItem: JBSTableItemDelegate) -> CGFloat static func tableView(tableView: UITableView, dequeueReusableCellWithIndexPath indexPath: NSIndexPath, tableItem: JBSTableItemDelegate, delegate: AnyObject) -> JBSTableViewCell } 

注意使用关键字“static”。 这些方法是UITableViewCell子类中的类方法,添加静态似乎是我需要做的来validation这些类符合,或者我明白。

当我使用第一个协议的代码看起来像这样,它编译:

 let tableViewCellInterface = tableItem!.tableViewCellInterface() 

这是调用这个方法(作为一个例子):

 func tableViewCellInterface() -> JBSTableViewCellInterface { return JBSLiteratureTableViewCell.self as! JBSTableViewCellInterface } 

这将返回单元格的类,例如“JBSLiteratureTableViewCell.self”

当我使用第二个协议的代码看起来像这样,它不会编译:

 returnFloat = tableViewCellInterface.tableView(tableView, heightForRowAtIndexPath: indexPath, tableItem: tableItem!) 

由于前面关于static关键字的原因,它无法编译,我得到的编译器错误是:

'JBSTableViewCellInterface'没有名为'tableView'的成员

如果我将静态关键字从协议funcs中取出,它会进行编译,但是UITableViewCell子类会抱怨说:

'JBSLiteratureTableViewCell'不符合协议'JBSTableViewCellInterface'

这是因为他们现在试图确保实例方法存在,而不是这些方法。

我如何使一个Swift类在类级别上符合协议,所以它可以是我的委托而不是类的某个实例? 我确信我可以通过创buildhelper类的协议JBSTableViewCellInterface是单身人士,让他们做的工作,但我宁愿build立它的类方法的UITableViewCell子类。

更新了Swift 2.0和更高版本

根据Gregzo的回答,Swift 2.0+允许在协议定义中将方法声明为静态。 这些必须满足实现协议的对象中的静态/类方法。

你不能满足一个静态方法的实例方法的协议定义,反之亦然,这使得上述问题的答案不完整。

如果你想试试这个,只需在你的协议定义中使用关键字“static”来实现静态方法或类方法在你的合规对象中实现的方法:

 protocol InstanceVsStatic { func someInstanceFunc() static func someStaticFunc() } enum MyConformingEnum: InstanceVsStatic { case someCase static func someStaticFunc() { // code } func someInstanceFunc() { // code } } class MyConformingClass: InstanceVsStatic { class func someStaticFunc() { // code } func someInstanceFunc() { // code } } struct MyConformingStruct: InstanceVsStatic { static func someStaticFunc() { // code } func someInstanceFunc() { // code } } 

你可以有一个实例方法调用静态/类方法:

这允许您在需要符合需要实例方法的协议时执行静态代码。

 struct MyConformingStruct: InstanceVsStatic { static func doStuffStatically(){ // code } static func someStaticFunc() { // code } func someInstanceFunc() { MyConformingStruct.doStuffStatically() } } 

Swift 1.2

除了间接地如上所述,在纯swift版本1.2及以下版本中,没有办法使用静态(类)方法来符合协议。 这是一个已知的bug /未实现的function: https : //openradar.appspot.com/20119848

如果协议func是静态的,那么实现者应该将其实现为静态方法,如果不是,则作为实例方法:

 protocol MixedProtocol { static func staticFoo() func instanceBar() } class ExampleClass : MixedProtocol { // implementing static func as class func is fine. // class funcs are overridable, not static ones class func staticFoo() { println( "I'm a class func" ) } func instanceBar() { println( "I'm an instance func" ) } } 

没有直接的方法:符合协议就意味着这一点,“静态”是协议成员声明的一个关键特征,实施者必须遵守。