21

基本上,有两件事我不明白:带有对象的对象和带有对象列表的对象

假设我从服务器收到一个对象列表。他们每个人看起来像这样:

@Entity
public class BigObject {
    @PrimaryKey
    private int id;
    private User user;
    private List<SmallObject> smallObjects;
}

将这两个对象作为字段:

@Entity
public class User {
    @PrimaryKey
    private int id;
    private String name;
    @TypeConverters(GenderConverter.class)
    public MyEnums.Gender gender;
}

@Entity
public class SmallObject {
    @PrimaryKey (autoGenerate = true)
    private int id;
    private String smallValue;
}

它们比这更复杂,所以我不能像 Room 建议的那样使用 @TypeConverters:

error: Cannot figure out how to save this field into database. You can consider adding a type converter for it.

如何将这个数据结构存储在 Room 中?

4

2 回答 2

38

我认为回答这个问题的最佳方法是简要概述存储结构......

列表

Room 不支持存储嵌套在 POJO 中的列表。存储列表的推荐方法是使用外键方法。将对象列表存储在单独的表(在本例中为 smallObjects 表)中,并带有指向其相关父对象(在本例中为“big_object_id”)的外键。它应该看起来像这样......

@Entity
public class BigObject {
    @PrimaryKey
    private int id;
    private User user;
    @Ignore
    private List<SmallObject> smallObjects;
}

@Entity(foreignKeys = {
            @ForeignKey(
                entity = BigObject.class,
                parentColumns = "id",
                childColumns = "big_object_fk"
            )})
public class SmallObject {
    @PrimaryKey (autoGenerate = true)
    private int id;
    private String smallValue;
    @ColumnInfo(name = "big_object_fk")
    private int bigObjectIdFk
}

请注意,我们添加了@Ignore注释,List<SmallObject>因为我们希望在房间持久性期间忽略该字段(因为不支持列表)。它现在存在,因此当我们从数据库请求相关小对象列表时,我们仍然可以将它们存储在 POJO 中。

据我所知,这意味着您要进行两个查询。

BigObject b = db.BigObjectDao.findById(bOId);
List<SmallObject> s = db.smallObjectDao.findAllSOforBO(bOId);
b.setsmallObjects(s);

似乎@Relation的形式对此有一个短手

类型转换器

这些适用于您拥有可以展平而不会丢失信息并存储在单个列中的复杂数据结构的情况。Date 对象就是一个很好的例子。Date 对象很复杂并且包含很多值,因此将其存储在数据库中很棘手。我们使用类型转换器来提取日期对象的毫表示并存储它。然后,我们在输出时将毫秒转换为日期对象,从而保持我们的数据完整。

嵌入式

当您想要获取父 POJO 中所有嵌套 POJO 的字段并将它们展平以存储在一个表中时,使用此方法。一个例子 :

- name
- age
- location
    - x 
    - y
- DOB

..嵌入时,此结构将存储在数据库中:

- name 
- age 
- location_x
- location_y
- DOB

从某种意义上说,Embedded 的存在是为了节省您为每个包含主要类型字段(如 String、int、float 等)的嵌套对象创建类型转换器的时间......

于 2017-07-03T16:03:23.667 回答
4

将 Object/ 转换List<Object>为 String,然后将其存储。

您可以将房间库中的对象存储为字符串。为此,您可以序列化对象并将其作为字符串存储在房间数据库中。

存储到房间

对象 -> 序列化 -> 字符串 -> 存储

从房间读取

字符串 -> 反序列化 -> 对象 -> 读取

如何序列化/反序列化?

有很多可用的选项。您可以手动完成,也可以为此使用库。您可以使用 Google 的GSON 库。它很容易使用。

代码:对象 -> 字符串

public String stringFromObject(List<YourClass> list){

    Gson gson = new Gson();
    String jsonString = gson.toJson(list);
    return jsonString;

}

代码:字符串-> 对象

public List<YourClass> getObjectFromString(String jsonString){

    Type listType = new TypeToken<ArrayList<YourClass>>(){}.getType();
    List<YourClass> list = new Gson().fromJson(jsonString, listType);
    return list;

}
于 2019-11-26T18:28:41.623 回答