我试图加载实体子列表,但我想避免做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
}
}
}
它无法正常工作,因为无法使用内部类作为转换器类。
RoomRegionDao.selectAll
时自动在区域实体中?答案 0 :(得分:3)
我认为-allApplications
仅适用于静态方法。
我根据here和here
来自关系部分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>,