0

我正在尝试创建一个嵌入式字段。这是一个简单的例子,但我无法让这个简单的例子工作。最终我需要有 3 个级别的嵌入式项目,但试图让这个测试用例工作。

@Entity(tableName = "userItemsEntity")
@Parcelize
data class Item(
    var objecttype: String?,
    @PrimaryKey(autoGenerate = false)
    var objectid: Int?,
    var subtype: String?,
    var collid: Int?,
    @Embedded
    var name: Name?
) : Parcelable

@Parcelize
data class Name(
    var primary: Boolean? = true,
    var sortindex: Int? = null,
    var content: String? = null) : Parcelable

当我尝试编译它时,它会在 DAO 上抱怨 updateItem()

SQL error or missing database (no such column: name) 

DAO 函数

@Query("UPDATE userItemsEntity SET " +
    "objecttype=:objecttype, objectid=:objectid, subtype=:subtype, collid=:collid, name=:name " +
    "WHERE objectid=:objectid")
fun updateItem(
    objecttype: String?,
    objectid: Int,
    subtype: String?,
    collid: Int?,
    name: Name?)
4

1 回答 1

1

原因是它说没有名称列。相反,该表由列组成,根据 EMBEDDED 类的成员变量(即主要、排序索引和内容)。

即表创建 SQL 是/将是:-

CREATE TABLE IF NOT EXISTS `userItemsEntity` (`objecttype` TEXT, `objectid` INTEGER, `subtype` TEXT, `collid` INTEGER, `primary` INTEGER, `sortindex` INTEGER, `content` TEXT, PRIMARY KEY(`objectid`))

Room 知道在提取行时从这些列构建相应的Name对象。

所以你可以使用: -

@Query("UPDATE userItemsEntity SET " +
        "objecttype=:objecttype, objectid=:objectid, subtype=:subtype, collid=:collid, `primary`=:primary, sortindex=:sortindex, content=:content " +
        "WHERE objectid=:objectid")
fun updateItem(
    objecttype: String?,
    objectid: Int,
    subtype: String?,
    collid: Int?,
    primary: Boolean?,
    sortindex: Int?,
    content: String?
)
  • 请注意,primary 是一个 SQLite 标记,因此包含在重音中以确保它不被视为标记。否则你会得到: -

    There is a problem with the query: [SQLITE_ERROR] SQL error or missing database (near "primary": syntax error)
    

但是,当您使用基于主键(objectid)的 WHERE 子句时,更新将仅适用于单行,因此您可以简单地使用:-

@Update
fun update(item: Item): Int
  • 显然函数的名称不需要更新它可以是任何适合的有效名称。
  • 这不仅具有更简单的优点,而且还可以返回更新的行数(如果行存在,则为 1,否则为 0)

实施名称列

如果您想要一个名称列并且该名称列包含一个名称对象。然后,由于 SQLite 没有对象的存储/列类型,因此您不会嵌入 Name 类。

您将var name: Name? 使用适当的 TypeConverter将 Name 对象转换为 SQLite 满足的类型:-

  • 文本(字符串),

  • REAL(浮点数,双精度...),

  • INTEGER (Long, Int ...) 或

  • BLOB(字节数组))。

  • 通常使用字符串,通常使用 GSON 将对象转换为 JOSN 字符串。

  • SQlite 确实有一个 NUMERIC 类型。但是,Room 不支持它的使用。我相信,因为其他类型涵盖了所有类型的数据,而 NUMERIC 是包罗万象的/默认值。

但是,使用对象的 JSON 表示会引入膨胀并降低从 SQL 方面转换的数据的有用性。

例如说你有: -

@Entity(tableName = "userOtherItemsEntity")
@Parcelize
data class OtherItem (
    var objecttype: String?,
    @PrimaryKey(autoGenerate = false)
    var objectid: Int?,
    var subtype: String?,
    var collid: Int?,
    var name: OtherName?) : Parcelable
@Parcelize
data class OtherName(
    var primary: Boolean? = true,
    var sortindex: Int? = null,
    var content: String? = null) : Parcelable

然后基础表确实有名称列。Room 生成的 CREATE SQL 将是:-

CREATE TABLE IF NOT EXISTS `userOtherItemsEntity` (`objecttype` TEXT, `objectid` INTEGER, `subtype` TEXT, `collid` INTEGER, `name` TEXT, PRIMARY KEY(`objectid`))

但是,您需要 TypeConverters 这可能是:-

@TypeConverter
fun fromOtherName(othername: OtherName ): String {
    return Gson().toJson(othername)
}
@TypeConverter
fun toOtherName(json: String): OtherName {
    return Gson().fromJson(json,OtherName::class.java)
}
  • 第一个使用 Gson 将对象转换为 JSON 字符串,例如在插入数据时
  • 第二个将 JSON 字符串转换为 OtherName 对象。

使用嵌入名称的项目,然后数据将按照以下方式存储:-

在此处输入图像描述

在转换带有 OtherName 的 OtherItem 时,数据(类似数据)将沿着以下行:-

在此处输入图像描述

  • 在前者中,3 个 Name 列将占用大约 (1 + 1 + 12) = 16 个字节。

  • 在后者中,OtherName 列(在使用时不考虑“Other”一词)将占用大约 55 个字节。

  • 如果要将 OtherName 的组件包含在搜索中,则后者可能需要更复杂且资源昂贵的搜索。

    • 例如,@Query("SELECT * FROM userItemsEntity WHERE 主要")的,而不是@Query("SELECT * FROM userOtherItemsEntity WHERE instr(name,'primary\":true') > 0")
于 2021-10-19T20:14:31.907 回答