我正在使用Room Library在我的Android App中保存数据。我有2个主表,分别是Task和Group。关系是一对多,其中一个任务只能属于一个组,但一个组可以属于许多任务。
持久性工作正常,但问题是当我尝试更新组中的信息时,与该组相关的每个任务都将被删除。
以下是我的实体和DAO配置:
@Entity(tableName = "task",
foreignKeys = {
@ForeignKey(
entity = Group.class,
parentColumns = "id",
childColumns = "groupId",
onDelete = CASCADE,
onUpdate = RESTRICT
)},
indices = {@Index(value = "id"), @Index(value = "groupId")}
)
public class Task {
@PrimaryKey(autoGenerate = true)
private int id;
private String name;
private int groupId;
public Task(int id, String name, int groupId) {
this.id = id;
this.name = name;
this.groupId = groupId;
}
}
@Entity(tableName = "group")
public class Group {
@PrimaryKey(autoGenerate = true)
private int id;
private String name;
public Group(int id, String name) {
this.id = id;
this.name = name;
}
}
@Dao
public interface TaskDao {
@Insert(onConflict = OnConflictStrategy.REPLACE)
void addTask(Task task);
@Query("select * from task")
public List<Task> listAllTasks();
@Query("select * from task where id = :taskId")
public Task getTask(int taskId);
@Update(onConflict = OnConflictStrategy.REPLACE)
void updateTask(Task task);
@Query("delete from task")
void removeAllTasks();
@Delete
void delete(Task task);
}
@Dao
public interface GroupDao {
@Insert(onConflict = OnConflictStrategy.REPLACE)
void addGroup(Group group);
@Query("select * from `group`")
public List<Group> listAllGroups();
@Query("select * from `group` where id = :groupId")
public Group getGroup(long groupId);
@Update(onConflict = OnConflictStrategy.REPLACE)
void updateGroup(Group group);
@Query("delete from `group`")
void removeAllGroups();
@Delete
void delete(Group group);
}
在Task类的Entity声明中将OnUpdate更改为RESTRICT,CASCADE或NO_ACTION并没有帮助。
我将不胜感激任何帮助。谢谢。
答案 0 :(得分:0)
As in SQLite Documentation: 当发生UNIQUE或PRIMARY KEY约束冲突时,REPLACE算法会在插入或更新当前行之前删除导致约束冲突的现有行,并且该命令将继续正常执行。如果发生NOT NOT NULL约束冲突,则REPLACE冲突解决方案将NULL值替换为该列的默认值,或者如果该列没有默认值,则使用ABORT算法。如果发生CHECK约束或外键约束冲突,则REPLACE冲突解决算法的工作方式类似于ABORT。
当REPLACE冲突解决策略为了满足约束条件而删除行时,当且仅当启用了递归触发器时,才会触发删除触发器。
And as in google Documentations: 默认情况下,所有RoomDatabase都在TEMP表的内存存储中使用并启用递归触发器。
结论:**由于您使用的是REPLACE
ConflictStrategy,并且外键** CASCADE 删除了父实体,因此当您尝试插入父行时,每个子行都会删除已经存在。
解决方案:
使用@Query
像这样更新组
@Query("UPDATE groups SET columnToUpdate1 = :value1, columnToUpdate2 = :value2 WHERE id=:id")
void updateGroup(int id, value1,value2);