我在整个应用程序中有几次调用,其中通过调用.createOrUpdate()
更新数据库中的表
此调用中的documentation suggests:
如果没有在数据库中创建项目,这是一种便捷的方法
存在。从数据参数中提取ID,然后按ID查询
在数据库上进行。如果数据库中具有相同ID的行
存在,那么数据库中的所有列将从
data参数中的字段。如果id为null(或0或一些
其他默认值)或数据库中不存在该对象
将在数据库中创建。这也意味着您的数据项
必须定义一个ID字段。
我的理解是,当调用createOrUpdate()时,如果数据库不包含该行,则底层代码应调用“插入”,如果该行存在,则应调用“更新”。
似乎正在发生的事情是,当该行已存在并导致异常时,底层代码正在调用“插入”。
有人知道为什么会这样或如何避免吗?
以下例外:
java.lang.RuntimeException: java.sql.SQLException: Unable to run insert stmt on object
Feed(id=126275579_organisations-email_58284484, type=organisations-email, category=1, msg=<p>Body</p>, regarding=Bobby Castle, subject=Subject, createdTs=2016-01-04T09:59:27+00:00, completed=0, read=0, starred=0, archived=0, remoteAttachments=[], localAttachments=null, authorId=250008275, authorName=Live Regression Test, authorImg=null): INSERT INTO `feed` (`id` ,`type` ,`category` ,`msg` ,`regarding` ,`subject` ,`createdTs` ,`completed` ,`read` ,`starred` ,`archived` ,`authorId` ,`authorName` ,`authorImg` ) VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?)
at com.j256.ormlite.dao.RuntimeExceptionDao.createOrUpdate(RuntimeExceptionDao.java:252)
at uk.co.test.test.data.orm.models.Feed.createOrUpdate(Feed.java:81)
at uk.co.test.test.interactors.server.CategoryInteractor$6.onSuccessfulResponse(CategoryInteractor.java:308)
at uk.co.test.test.interactors.server.CategoryInteractor$6.onSuccessfulResponse(CategoryInteractor.java:290)
at uk.co.test.test.interactors.common.StandardCallbackRunnable.checkStandardResponse(StandardCallbackRunnable.java:27)
at uk.co.test.test.interactors.common.StandardCallbackRunnable.checkStandardResponse(StandardCallbackRunnable.java:12)
at uk.co.test.test.interactors.common.CallbackRunnable.success(CallbackRunnable.java:127)
at retrofit.CallbackRunnable$1.run(CallbackRunnable.java:45)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1112)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:587)
at java.lang.Thread.run(Thread.java:818)
Caused by: java.sql.SQLException: Unable to run insert stmt on object Feed(id=126275579_organisations-email_58284484, type=organisations-email, category=1, msg=<p>Body</p>, regarding=Bobby Castle, subject=Subject, createdTs=2016-01-04T09:59:27+00:00, completed=0, read=0, starred=0, archived=0, remoteAttachments=[], localAttachments=null, authorId=250008275, authorName=Live Regression Test, authorImg=null): INSERT INTO `feed` (`id` ,`type` ,`category` ,`msg` ,`regarding` ,`subject` ,`createdTs` ,`completed` ,`read` ,`starred` ,`archived` ,`authorId` ,`authorName` ,`authorImg` ) VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?)
at com.j256.ormlite.misc.SqlExceptionUtil.create(SqlExceptionUtil.java:22)
at com.j256.ormlite.stmt.mapped.MappedCreate.insert(MappedCreate.java:135)
at com.j256.ormlite.stmt.StatementExecutor.create(StatementExecutor.java:450)
at com.j256.ormlite.dao.BaseDaoImpl.create(BaseDaoImpl.java:310)
at com.j256.ormlite.dao.BaseDaoImpl.createOrUpdate(BaseDaoImpl.java:336)
at com.j256.ormlite.dao.RuntimeExceptionDao.createOrUpdate(RuntimeExceptionDao.java:249)
at uk.co.test.test.data.orm.models.Feed.createOrUpdate(Feed.java:81)
at uk.co.test.test.interactors.server.CategoryInteractor$6.onSuccessfulResponse(CategoryInteractor.java:308)
at uk.co.test.test.interactors.server.CategoryInteractor$6.onSuccessfulResponse(CategoryInteractor.java:290)
at uk.co.test.test.interactors.common.StandardCallbackRunnable.checkStandardResponse(StandardCallbackRunnable.java:27)
at uk.co.test.test.interactors.common.StandardCallbackRunnable.checkStandardResponse(StandardCallbackRunnable.java:12)
at uk.co.test.test.interactors.common.CallbackRunnable.success(CallbackRunnable.java:127)
at retrofit.CallbackRunnable$1.run(CallbackRunnable.java:45)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1112)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:587)
at java.lang.Thread.run(Thread.java:818)
Caused by: java.sql.SQLException: inserting to database failed: INSERT INTO `feed` (`id` ,`type` ,`category` ,`msg` ,`regarding` ,`subject` ,`createdTs` ,`completed` ,`read` ,`starred` ,`archived` ,`authorId` ,`authorName` ,`authorImg` ) VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?)
at com.j256.ormlite.misc.SqlExceptionUtil.create(SqlExceptionUtil.java:22)
at com.j256.ormlite.android.AndroidDatabaseConnection.insert(AndroidDatabaseConnection.java:169)
at com.j256.ormlite.stmt.mapped.MappedCreate.insert(MappedCreate.java:91)
at com.j256.ormlite.stmt.StatementExecutor.create(StatementExecutor.java:450)
at com.j256.ormlite.dao.BaseDaoImpl.create(BaseDaoImpl.java:310)
at com.j256.ormlite.dao.BaseDaoImpl.createOrUpdate(BaseDaoImpl.java:336)
at com.j256.ormlite.dao.RuntimeExceptionDao.createOrUpdate(RuntimeExceptionDao.java:249)
at uk.co.test.test.data.orm.models.Feed.createOrUpdate(Feed.java:81)
at uk.co.test.test.interactors.server.CategoryInteractor$6.onSuccessfulResponse(CategoryInteractor.java:308)
at uk.co.test.test.interactors.server.CategoryInteractor$6.onSuccessfulResponse(CategoryInteractor.java:290)
at uk.co.test.test.interactors.common.StandardCallbackRunnable.checkStandardResponse(StandardCallbackRunnable.java:27)
at uk.co.test.test.interactors.common.StandardCallbackRunnable.checkStandardResponse(StandardCallbackRunnable.java:12)
at uk.co.test.test.interactors.common.CallbackRunnable.success(CallbackRunnable.java:127)
at retrofit.CallbackRunnable$1.run(CallbackRunnable.java:45)
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1112)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:587)
at java.lang.Thread.run(Thread.java:818)
Caused by: android.database.sqlite.SQLiteConstraintException: UNIQUE constraint failed: feed.id (code 1555)
at android.database.sqlite.SQLiteConnection.nativeExecuteForLastInsertedRowId(Native Method)
at android.database.sqlite.SQLiteConnection.executeForLastInsertedRowId(SQLiteConnection.java:782)
at android.
最佳答案
我的理解是,当调用createOrUpdate()时,如果数据库不包含该行,则底层代码应调用“插入”,如果该行存在,则应调用“更新”。
这是正确的,但是该操作不是(也不能是)原子的,因为要进行多个数据库调用才能完成该操作。 DAO方法中没有完成锁定-尽管也许应该这样做。
如果我们查看BaseDaoImpl.createOrUpdate(...)
code,您会发现它确实存在:
从相关实体中提取ID。
在数据库中查找它是否存在。
如果update(...)
存在,则调用create()
,否则createIfNotExists(...)
。
因为有多个对数据库的调用正在进行,所以如果多个线程同时执行此方法,则存在竞争条件。
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1112)
正确,如果您将此DAO方法与线程一起使用(并且
synchronized
也会执行2次数据库调用),那么我将在块内执行此操作:synchronized (dao) {
status = dao.createOrUpdate(...);
}
关于java - 使用Ormlite createOrUpdate()方法获取“唯一约束失败”异常,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/34588986/