简体   繁体   English

具有通用数据类型的 Swift 下标

[英]Swift subscript with generic data type

I am trying to program a two-dimensional data storage struct for various data types.我正在尝试为各种数据类型编写二维数据存储结构。 However, I am struggling with the subscript for setting the data due 'Cannot assign value of type 'T' to subscript of type 'T' errors.但是,由于“无法将'T'类型的值分配给'T'类型的下标错误,我正在努力设置数据的下标。 Any help is much appreciated!任何帮助深表感谢!

struct dataMatrix<T> : Sequence, IteratorProtocol {
    var rows: Int, columns: Int
    var data: [T]
    var position = 0
    
    init(rows: Int, columns: Int) {
        self.rows = rows
        self.columns = columns
        data = Array<T>()
    }
    
    func valueAt(column: Int, row: Int) -> T? {
        guard column >= 0 && row >= 0 && column < columns else {
                return nil
            }

        let indexcolumn = column + row * columns

            guard indexcolumn < data.count else {
                return nil
            }

            return data[indexcolumn]
        }
    }
    
    subscript<T>(column: Int, row:Int) -> T?{
        get{
            return valueAt(column: column, row: row) as? T
        }
        set{
            data[(column * row) + column] = (newValue as! T) // does not compile
        }   
    }
    
    // sequence iterator protorocl methods
    mutating func next() -> String? {
        if position <= data.count{
            print(position)
            defer { position += 1 }
            return "\(position)"
        }else{
            defer {position = 0}
            return nil
        }
    }
}
subscript<T>(column: Int, row:Int) -> T?{

defines a generic method with a type placeholder T which is unrelated to the generic type T of struct dataMatrix<T> .定义了一个具有类型占位符T的泛型方法,它与struct dataMatrix<T>的泛型类型T无关。 The solution is simple: Remove the type placeholder:解决方案很简单:删除类型占位符:

subscript(column: Int, row: Int) -> T? {
    // ...
}

That makes also the type casts inside the getter and setter unnecessary.这也使得 getter 和 setter 内部的类型转换变得不必要。 You only have to decide what to do if the setter is called with a nil argument (eg: nothing):如果使用nil参数调用 setter(例如:无),您只需决定要做什么:

subscript(column: Int, row: Int) -> T? {
    get {
        return valueAt(column: column, row: row)
    }
    set {
        if let value = newValue {
            data[(column * row) + column] = value
        }
    }
}

Another option is to make the return type of the subscript method non-optional, and treat invalid indices as a fatal error (which is how the Swift Array handles it):另一种选择是使下标方法的返回类型不可选,并将无效索引视为致命错误(这是 Swift Array处理它的方式):

subscript(column: Int, row: Int) -> T {
    get {
        guard let value = valueAt(column: column, row: row) else {
            fatalError("index out of bounds")
        }
        return value
    }
    set {
        data[(column * row) + column] = newValue
    }
}

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM