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

时间:2017-04-13 09:26:37

标签: ios swift sqlite swift3 sqlite.swift

我已经创建了一个Database类:

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)
}

它不起作用。它没有意义,因为我也尝试在事务中创建表并且运行良好。

非常感谢任何帮助!

1 个答案:

答案 0 :(得分:0)

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

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