首页 > 解决方案 > Swift 中的 GRDB - .fetchAll 只返回一列

问题描述

我的原始代码

func getUserByEmpNum(_ id: Int) -> String {
    let nameQuery: String = "SELECT fld_str_firstname, fld_str_lastName FROM userView WHERE fld_int_id = \(id);"

    var returnStr = ""
    do {
        let dbQueue = try DatabaseQueue(path: MYCDatabase.pathToDatabase)
        try dbQueue.inTransaction(.none) { (db) -> Database.TransactionCompletion in
            let returnStrs = try String.fetchAll(db, sql: nameQuery)
            // Placing a breakpoint here, returnStrs only has one element?
            return .commit
        }
    } catch {
        print (error)
    }

    return returnStr
}

我的问题

在这段代码中,如果我执行类似的查询,select fld_str_firstname from myOwnUserView where fld_int_id = 2;我会在 returnStrs 数组中得到一个元素,这与预期的一样。然后选择这两个字段,就像在 nameQuery 中一样,我仍然只能在 returnStrs 数组中得到一个字符串。

为什么会这样,我如何适应它以获取响应中的所有选定列?

标签: swiftgrdb

解决方案


String.fetchAll返回从最左侧选定列中提取的字符串数组,如文档所示。每个提取的行一个字符串。每个选定的列没有一个字符串。

如果您想从多列中获取字符串,请使用Row.fetchAll,它返回一个数据库行数组。从这些行中,您可以提取您感兴趣的每一列:

let rows = try Row.fetchAll(db, sql: "SELECT fld_str_firstname, fld_str_lastName FROM ...")
for row in rows {
    let firstName: String = row["fld_str_firstname"]
    let lastName: String = row["fld_str_lastName"]
}

有关从数据库行中提取值的更多信息,请参阅文档的这一章。

由于您正在从以其 id 标识的单行中读取名称,因此您可能更喜欢fetchOne使用单个数据库行的方法(请参阅Fetching Methods):

if let row = try Row.fetchOne(db, sql: "SELECT ... WHERE fld_int_id = ?", arguments: [id]) {
    let firstName: String = row["fld_str_firstname"]
    let lastName: String = row["fld_str_lastName"]
    // Use first and last name 
} else {
    // ID does not exist in the database: do what is appropriate.
}

推荐阅读