使用事务插入会抛出错误 Sqlite.swift



我创建了一个数据库类,如下所示:

class Database {
    static let instance = Database()
    private let categories = Table("Category")
    private var db: Connection?
    let cat_id = Expression<String>("id")
    let cat_name = Expression<String>("name")

    private init() {
        let path = NSSearchPathForDirectoriesInDomains(
            .documentDirectory, .userDomainMask, true
            ).first!
        do {
            db = try Connection("(path)/SalesPresenterDatabase.sqlite3")
            createTable()
        } catch {
            print("error")
        }
    }
    func createTable() {
        do{
            try self.db!.run(self.categories.create(ifNotExists: true) { table in
                table.column(self.cat_id)
                table.column(self.cat_name)
            })
        }catch{
             print("error")
        }
    }
    func addRow(table: DBTableNames, object: [Any]) -> Int64? {
      do {
         try self.db!.transaction() {
            for obj in object{
                if table.rawValue == DBTableNames.Category.rawValue{
                    let cats : CategoryObject = obj as! CategoryObject
                    let insert = self.categories.insert(self.cat_id <- cats.id,
                                                        self.cat_name <- cats.name)
                    try self.db!.run(insert)
                }
             }
          }
        }catch{
           print("Insert failed (error)")
           return -1
        }
        return 1
         }
}

然后,我继续调用我的代码以通过运行以下命令来添加行:

    let returnValue = Database.instance.addRow(table: DBTableNames(rawValue: entity)!,
                                               object: databaseObject)

我遇到的问题是它总是抛出一个错误说:

插入失败 无法完成操作。(SQLite.Result 错误 0.(和完整:无法回滚 - 没有事务处于活动状态(代码: 1(

如果我再次看到此错误,我的Mac将退出窗口!

整个操作都在后台线程中,但我也尝试了以下内容:

    DispatchQueue.main.async{
    let returnValue = Database.instance.addRow(table: DBTableNames(rawValue: entity)!,
                                                   object: databaseObject)
}

它没有用。这没有意义,因为我也尝试在事务中创建表并且效果很好。

任何帮助将不胜感激!

您的func createTable()func addRow()方法不是线程安全的。多个线程可以同时访问它。

在 Singleton 类中创建一个私有串行DispatchQueue,并通过此串行队列执行上述功能。这将防止同时从多个线程访问数据库,并且串行队列将排队并发任务。

相关内容

  • 没有找到相关文章

最新更新