在超类中实现CodingKey映射

时间:2019-04-23 13:31:07

标签: swift struct swift-protocols

我有一个结构,该结构具有一些属性,一个CodingKey枚举(为了符合Codable协议)和一些计算变量,它们返回映射到字符串的键数组。

我想做的是将这些计算变量提取到协议中(如果我必须将结构更改为类,则为超类),以避免在每个结构/类中重复执行。我需要其中的很多

问题是我找不到找到CodingKey枚举的方法。

编辑: DBentity

/// Basic protocol for any object directly related with a database table.
public protocol DBEntity: Codable, Equatable {}

代码:

///////////////////////////////////////////////////////////////////////

// MARK: - Entity Properties

struct AccountsAddresses: DBEntity {

    let statusMobile: String?
    let codeAccount: Int?
    let codeUnitOrg: String?
    let codeSalesOrg: String?
    let codeAddress: Int?
    let codeType: String?
    let byDefault: String?
    let transfered: String?
}

///////////////////////////////////////////////////////////////////////

// MARK: - Table Columns

extension AccountsAddresses {

    /// Table name.
    static var tableName = "ACCOUNTS_ADDRESSES"

    /// Table columns keys.
    enum CodingKeys: String, CodingKey, CaseIterable {
        case statusMobile   = "status_mobile"
        case codeAccount    = "code_account"
        case codeUnitOrg    = "code_unit_org"
        case codeSalesOrg   = "code_sales_org"
        case codeAddress    = "code_address"
        case codeType       = "code_type"
        case byDefault      = "by_default"
        case transfered
    }

/* This is what I'm trying to extract to a protocol(extension) or super-class. */

    /// All table columns keys.
    static var columns: [String] = CodingKeys.allCases.map { $0.rawValue }
    static var columnsJoined: String = columns.joined(separator: String.commaSpace)
    static var columnsTableName: [String] = columns.map { tableName + String.dot + $0 }
    static var columnsJoinedTableName: String = columnsTableName.joined(separator: String.commaSpace)
}

编辑

使用Sweeper答案中的给定代码,我试图在新的结构中实现该协议,但是编译器要求将类型设置为typealias CodingKeyType(因为定义的协议具有关联类型)。

public protocol Table {
    associatedtype ColumnKeysType: (CodingKey & CaseIterable & RawRepresentable)
    static var tablename: String { get }
}

这里是要测试的结构:

struct AccountsTest: Table {

    typealias CodingKeyType = <#type#>
}

我试图制作另一个结构以便为Typealias分配类型,但是我缺少/做错了一些事情(我必须实现此var,构造函数和typealias)。

struct Keys: (CodingKey & CaseIterable & RawRepresentable) {

    var stringValue: String
    var rawValue: String
    var intValue: Int?

    init?(stringValue: String) {}
    init?(rawValue: String) {}
    init?(intValue: Int) {}

    typealias AllCases = Keys.RawValue
    typealias RawValue = String
}

编译器继续显示错误,我找不到实现此目的的方法。

'CaseIterable' requires the types 'Keys' and 'String.Element' (aka 'Character') be equivalent

完成这项工作的任何提示吗?

谢谢

2 个答案:

答案 0 :(得分:0)

请注意,您尝试提取的属性不是计算的属性。

您要提取的属性似乎仅取决于tableNameCodingKeys,因此我们可以编写如下协议:

protocol SomeProtocol {
    associatedtype CodingKeyType : (CodingKey & CaseIterable & RawRepresentable)
    static var tableName: String { get }
}

然后是这样的扩展名(我已将属性转换为计算的属性):

extension SomeProtocol where CodingKeyType.RawValue == String {
    static var columns: [String] { return CodingKeyType.allCases.map { $0.rawValue } }
    static var columnsJoined: String { return columns.joined(separator: " ") }
    static var columnsTableName: [String] { return columns.map { tableName + "." + $0 } }
    static var columnsJoinedTableName: String { return columnsTableName.joined(separator: " ") }
}

答案 1 :(得分:-1)

您不需要自己进行转换。您可以使用decoder.keyDecodingStrategy = .convertFromSnakeCase,并且您的json将映射到您的骆驼保护套属性。

有关更多信息:

https://developer.apple.com/documentation/foundation/jsondecoder/keydecodingstrategy/convertfromsnakecase