我创建了一个数据库类,如下所示:
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
,并通过此串行队列执行上述功能。这将防止同时从多个线程访问数据库,并且串行队列将排队并发任务。