美文网首页Swift算法
Swift数据结构-2D数组 Array2D

Swift数据结构-2D数组 Array2D

作者: UnsanYL | 来源:发表于2017-07-02 23:34 被阅读27次

    声明:算法和数据结构的文章均是作者从github上翻译过来,为方便大家阅读。如果英语阅读能力强的朋友,可以直接到swift算法俱乐部查看所有原文,以便快速学习。作者同时也在学习中,欢迎交流

    在C和Objective-C中,我们会写以下代码来声明一个9x7的网格组合:

    int cookies[9][7];

    这个代码会创建出一个含有63个元素的二位数组。如果要找出第3列第6行的网格,可以用一下方式:

    myCookie = cookies[3][6];

    而在swift中,这样创建多维数组的表达方式是不允许的。如果想要用swift创建一个多维数组,过程如下:

    var cookies = [[Int]]()
    for _ in 1...9 {
      var row = [Int]()
      for _ in 1...7 {
        row.append(0)
      }
      cookies.append(row)
    }
    

    然后,如果要找出第3列第6行的网格,可以用以下方式:

    let myCookie = cookies[3][6]

    我们也可以用单行代码的形式来创建该数组:

    var cookies = [[Int]](repeating: [Int](repeating: 0, count: 7), count: 9)

    可能这里看起来整个函数有点复杂,我们可以用一个辅助函数来简化一下:

    func dim<T>(_ count: Int, _ value: T) -> [T] {
      return [T](repeating: value, count: count)
    }
    

    简化后效果如下:

    var cookies = dim(9, dim(7, 0))

    这里swift会推断数组的数据类型为int,因为我们的初始数值都设定为0. 如果想要修改数据类型,可以在声明中直接修改:

    var cookies = dim(9, dim(7, "yum"))

    这里的dim()函数让我们可以随意创造出多维的数组,比如三维:

    var threeDimensions = dim(2, dim(3, dim(4, 0)))

    但是这样的创建方式有很明显的缺点,即我们无法知道每个维度代表什么信息。所以,我们可以自己创造自己的类型来达到同样的效果,同时又更加简单实用。

    public struct Array2D<T> {
      public let columns: Int
      public let rows: Int
      fileprivate var array: [T]
      
      public init(columns: Int, rows: Int, initialValue: T) {
        self.columns = columns
        self.rows = rows
        array = .init(repeating: initialValue, count: rows*columns)
      }
      
      public subscript(column: Int, row: Int) -> T {
        get {
          precondition(column < columns, "Column \(column) Index is out of range. Array<T>(columns: \(columns), rows:\(rows))")
          precondition(row < rows, "Row \(row) Index is out of range. Array<T>(columns: \(columns), rows:\(rows))")
          return array[row*columns + column]
        }
        set {
          precondition(column < columns, "Column \(column) Index is out of range. Array<T>(columns: \(columns), rows:\(rows))")
          precondition(row < rows, "Row \(row) Index is out of range. Array<T>(columns: \(columns), rows:\(rows))")
          array[row*columns + column] = newValue
        }
      }
    }
    

    Array2D是一个通类,它可以持有任意类型的对象,不单单数字。我们可以用以下方式创建Array2D实例:

    var cookies = Array2D(columns: 9, rows: 7, initialValue: 0)

    通过subscript函数,你可以用以下方式获取数组中的某个指定对象:

    let myCookie = cookies[column, row]

    或修改数组中的某个对象:

    cookies[column, row] = newCookie

    从原理来说,Array2D是用一个一维的数组来储存数据。特定对象的索引是通过(row x numberOfColumns) + column来换算获取,但是对于使用者来说,我们只要考虑这里的行和列就行,剩下的工作交给Array2D去完成。这也是将原始数据包装到类或者结构体里面的优点。

    相关文章

      网友评论

        本文标题:Swift数据结构-2D数组 Array2D

        本文链接:https://www.haomeiwen.com/subject/hrvocxtx.html