Room Persistence 在更新时删除子项

Her*_*lva 10 sqlite android persistence relationship android-room

我正在使用 Room Library 在我的 Android 应用程序中保存数据。我有 2 个主表,分别是 Task 和 Group。关系是一对多的,其中一个任务只能属于 1 个组,但一个组可以属于多个任务。

持久性工作正常,但问题是当我尝试从组更新信息时,与该组相关的每个任务都被删除。

下面是我的实体和 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;
    }

}
Run Code Online (Sandbox Code Playgroud)

集团实体

 @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;
    }

}
Run Code Online (Sandbox Code Playgroud)

任务DAO

@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);
}
Run Code Online (Sandbox Code Playgroud)

组DAO

@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);

}
Run Code Online (Sandbox Code Playgroud)

在 Task 类中的实体声明中将 OnUpdate 更改为 RESTRICT、CASCADE 或 NO_ACTION 没有帮助。

我将不胜感激任何帮助。谢谢你。

小智 6

如 SQLite 文档中所示: 当发生 UNIQUE 或 PRIMARY KEY 约束冲突时,REPLACE 算法在插入或更新当前行之前删除导致约束冲突的预先存在的行,并且命令继续正常执行。如果发生 NOT NULL 约束冲突,REPLACE 冲突解决方案会用该列的默认值替换 NULL 值,或者如果该列没有默认值,则使用 ABORT 算法。如果发生 CHECK 约束或外键约束冲突,REPLACE 冲突解决算法的工作方式类似于 ABORT。

当 REPLACE 冲突解决策略删除行以满足约束时,当且仅当启用递归触发器时才会触发删除触发器。

就像在 google 文档中一样: 默认情况下,所有 RoomDatabases 都在内存存储中用于 TEMP 表并启用递归触发器。

结论:**由于您使用的是REPLACEConflictStrategy,并且外键 **CASCADE将删除到父实体,因此当您尝试插入父行时,每个子行都将删除已存在。

解决方案: 使用@Query用于更新组这样

@Query("UPDATE groups SET columnToUpdate1 = :value1, columnToUpdate2 = :value2 WHERE id=:id")
void updateGroup(int id, value1,value2);
Run Code Online (Sandbox Code Playgroud)