一尘不染

如何在Swift中扩展类型化数组?

swift

如何扩展Swift Array<T>T[]带有自定义功能工具的类型?

浏览Swift的API文档可发现Array方法是的扩展T[],例如:

extension T[] : ArrayType {
    //...
    init()

    var count: Int { get }

    var capacity: Int { get }

    var isEmpty: Bool { get }

    func copy() -> T[]
}

当复制和粘贴相同的源并尝试任何变体时,例如:

extension T[] : ArrayType {
    func foo(){}
}

extension T[] {
    func foo(){}
}

它无法生成并显示以下错误:

标称类型T[]不能扩展

使用完整类型定义失败Use of undefined type 'T',即:

extension Array<T> {
    func foo(){}
}

并且也无法使用Array<T : Any>Array<String>

奇怪的是,Swift让我扩展了一个无类型数组:

extension Array {
    func each(fn: (Any) -> ()) {
        for i in self {
            fn(i)
        }
    }
}

它让我打电话给:

[1,2,3].each(println)

但是我无法创建适当的泛型类型扩展,因为当类型流过该方法时,该类型似乎丢失了

extension Array {
    func find<T>(fn: (T) -> Bool) -> T[] {
        var to = T[]()
        for x in self {
            let t = x as T
            if fn(t) {
                to += t
            }
        }
        return to
    }
}

但是编译器将其视为无类型的,但仍允许使用以下命令调用扩展名:

["A","B","C"].find { $0 > "A" }

当使用调试器进行步进指示类型Swift.String为时,尝试像字符串一样访问它而不String首先将其强制转换为构造错误,即:

["A","B","C"].find { ($0 as String).compare("A") > 0 }

有谁知道创建像内置扩展一样的类型化扩展方法的正确方法是什么?


阅读 257

收藏
2020-07-07

共1个答案

一尘不染

对于使用 扩展类型数组,以下内容对我有用(Swift 2.2 )。例如,对类型化数组进行排序:

class HighScoreEntry {
    let score:Int
}

extension Array where Element == HighScoreEntry {
    func sort() -> [HighScoreEntry] {
      return sort { $0.score < $1.score }
    }
}

尝试使用 structtypealias 执行此 操作 将产生错误:

Type 'Element' constrained to a non-protocol type 'HighScoreEntry'

更新

要使用 非类 扩展类型化数组,请使用以下方法:

typealias HighScoreEntry = (Int)

extension SequenceType where Generator.Element == HighScoreEntry {
    func sort() -> [HighScoreEntry] {
      return sort { $0 < $1 }
    }
}

Swift 3中, 某些类型已重命名:

extension Sequence where Iterator.Element == HighScoreEntry 
{
    // ...
}
2020-07-07