Swift 2:UITableViewDataSource协议扩展

时间:2015-09-15 09:36:41

标签: ios uitableview swift2 protocol-extension

我一直在玩协议扩展,我遇到了问题。也许我想要实现的目标无法实现。我有这个游乐场:

//: Playground - noun: a place where people can play

import UIKit

protocol ArrayContainer {
    typealias T
    var array: [T] { get }
}

class MyViewController: UIViewController, ArrayContainer, UITableViewDataSource {
    typealias T = String
    var array = ["I am", "an Array"] 
}

extension UITableViewDataSource where Self: ArrayContainer {

    func numberOfSectionsInTableView(tableView: UITableView) -> Int {
        return 1
    }

    func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return array.count
    }

    func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
        // Whatever
        return UITableViewCell()
    }   
}

这就是我拥有的和我想要的东西:

  • 我有一个协议ArrayContainer,它只有一个类型和一个包含此类型对象类型的数组
  • 我有一个UITableViewDataSource的协议扩展名,当该类符合ArrayController协议时使用。这只是将数组的项数作为行数返回。 cellForRowAtIndexPath方法没有很好地实现,但问题并非如此。
  • 我有一个名为UIViewController的{​​{1}}子类,它实现了两种协议。

问题是编译器抱怨因为MyViewController不符合MyViewController,但据我所知,它应该被UITableViewDataSource扩展覆盖。我在这里错过了什么吗?或者Objective-C协议可能无法扩展?

1 个答案:

答案 0 :(得分:4)

我知道回复有点迟,你甚至可能都没有找到这个答案,但我只是遇到了这个问题,需要一个现实世界的“解决方案”。您可以在类中实现UITableViewDataSource方法,然后立即将工作交给协议扩展,如下例所示。如果swift进行了不再需要的改进,那么更改回原始帖子中的代码很简单。

//: Playground - noun: a place where people can play

import UIKit

protocol ArrayContainer {
    associatedtype T
    var array: [T] { get }
}

class MyViewController: UIViewController, ArrayContainer, UITableViewDataSource {
    typealias T = String
    var array = ["I am", "an Array"]

    func numberOfSectionsInTableView(tableView: UITableView) -> Int {
        return self.internal_numberOfSectionsInTableView(tableView)
    }

    func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return self.internal_tableView(tableView, numberOfRowsInSection: section)
    }

    func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
        return self.internal_tableView(tableView, cellForRowAtIndexPath: indexPath)
    }
}

extension UITableViewDataSource where Self: ArrayContainer {

    func internal_numberOfSectionsInTableView(tableView: UITableView) -> Int {
        return 1
    }

    func internal_tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return array.count
    }

    func internal_tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
        // Whatever
        return UITableViewCell()
    }   
}