Android Room中的实体列表关系

时间:2017-05-27 13:09:20

标签: android database orm kotlin android-room

我试图加载实体子列表,但我想避免做2次查询。

我正在考虑在TypeConverter中进行查询,但我真的不知道这是不是一个好主意。

我的实体:

@Entity
class Region(
        @PrimaryKey(autoGenerate = true)
        var id: Int = 0,
        var name: String = "",
        var locales: List<Locale> = listOf())

@Entity(foreignKeys = arrayOf(ForeignKey(
        entity = Region::class,
        parentColumns = arrayOf("id"),
        childColumns = arrayOf("regionId"),
        onDelete = CASCADE,
        onUpdate = CASCADE
)))
class Locale(
        @PrimaryKey(autoGenerate = true)
        var id: Int = 0,
        var regionId: Int = 0,
        var name: String = "")

DAOs:

@Dao
interface RoomRegionDao{
    @Insert
    fun insert(region: Region)

    @Delete
    fun delete(region: Region)

    @Query("select * from region")
    fun selectAll(): Flowable<List<Region>>
}

@Dao
interface RoomLocaleDao{
    @Insert
    fun insert(locale: Locale)

    @Query("select * from locale where regionId = :arg0")
    fun selectAll(regionId: Int): List<Locale>
}

数据库:

@Database(entities = arrayOf(Region::class, Locale::class), version = 1)
@TypeConverters(RoomAppDatabase.Converters::class)
abstract class RoomAppDatabase : RoomDatabase() {
    abstract fun regionDao(): RoomRegionDao
    abstract fun localeDao(): RoomLocaleDao

    inner class Converters {
        @TypeConverter
        fun toLocales(regionId: Int): List<Locale> {
            return localeDao().selectAll(regionId)
        }

        @TypeConverter
        fun fromLocales(locales: List<Locale>?): Int {
            locales ?: return 0
            if (locales.isEmpty()) return 0

            return locales.first().regionId
        }
    }
}

它无法正常工作,因为无法使用内部类作为转换器类。

  • 这是一个好方法吗?
  • 如何加载&#34;区域设置列表&#34;我RoomRegionDao.selectAll时自动在区域实体中?

3 个答案:

答案 0 :(得分:3)

我认为-allApplications仅适用于静态方法。 我根据herehere

中的示例说出这一点

来自关系部分here

  

“因为SQLite是一个关系数据库,你可以指定对象之间的关系。即使大多数ORM库允许实体对象相互引用,Room也明确禁止它。”

所以我想最好在TypeConverter属性上添加@Ignore并在locales上创建一个插入RoomLocaleDao的方法并在插入区域后调用它。

插入Region的方法可以返回inserted id

  

如果@Insert方法只接收1个参数,则它可以返回long,即插入项的新rowId。如果参数是数组或集合,则应返回long []或List。   (https://developer.android.com/topic/libraries/architecture/room.html#daos-convenience

答案 1 :(得分:0)

只需一个Regions,就可以在每个Locales中获得Region的所有Query清单。

您需要做的就是创建一个新的Relation,如下所示:

data class RegionWithLocale(
    @Embedded
    val region: Region,
    @Relation(
        parentColumn = "id",
        entity = Locale::class,
        entityColumn = "regionId"
    )
    val locales: List<Locale>
)

然后后者在您的Dao中获得这种关系:

@Query("SELECT * FROM region WHERE id IS :regionId")
fun getRegions(regionId: Int): LiveData<RegionWithLocale>

@Query("SELECT * FROM region")
fun getAllRegions(): LiveData<List<RegionWithLocale>>

答案 2 :(得分:0)

此解决方案对我有用

ListConvert.kt

class ListConverter
{
    companion object
    {
        var gson = Gson()

        @TypeConverter
        @JvmStatic
        fun fromTimestamp(data: String?): List<Weather>?
        {
            if (data == null)
            {
                return Collections.emptyList()
            }

            val listType = object : TypeToken<List<String>>() {}.type
            return gson.fromJson(data, listType)
        }

        @TypeConverter
        @JvmStatic
        fun someObjectListToString(someObjects: List<Weather>?): String?
        {
            return gson.toJson(someObjects)
        }
    }
}

实体内部的属性

@SerializedName("weather")
@TypeConverters(ListConverter::class)
val weather: List<Weather>,