在玩房间持久性库时,我发现没有方法可以使用NOT NULL和UNIQUE约束设置数据类字段。 SQLite是否支持这些约束。迁移使用这些约束的旧数据库不是一个问题吗?任何人都可以就此问题提出建议吗?
答案 0 :(得分:35)
我开始知道没有方法可以设置带有NOT NULL和UNIQUE约束的数据类字段
@NonNull
字段上的@Entity
注释会导致该字段的列应用NOT NULL
。
unique=true
上的 @Index
会强制执行唯一性约束(例如@Entity(indices={@Index(value="something", unique=true)}
)。但是,您是正确的,不支持列上的明确UNIQUE
约束,而不是通过索引。
迁移使用这些约束的旧数据库是不是有问题?
Room不支持现有的数据库结构,特别是在当前的alpha
状态。随着时间的推移,我希望Room能够支持更高比例的SQLite功能,但如果它达到100%,我会感到震惊。
答案 1 :(得分:3)
关于使用Kotlin的人NOT NULL
的补充答案:
请注意,将类型标记为非可选将自动使其不为空(并且可选类型不会执行此操作)。
您可以在数据库中@Database(exportSchema = true)
的房间生成的架构中进行检查。
例如,我有类似的东西:
@Entity(tableName = "messages")
data class Message (
@PrimaryKey
val messageId: UUID = UUID.randomUUID(),
val date: Date = Date(),
val receivedDate: Date? = null
)
在生成的架构中,我可以阅读:
"CREATE TABLE IF NOT EXISTS `${TABLE_NAME}` (`messageId` TEXT NOT NULL, `date` INTEGER NOT NULL, `receivedDate` INTEGER, PRIMARY KEY(`messageId`))"
(注意:日期类型在这里是Int
,UUID是由于我在其他地方使用的转换器而产生的字符串)
答案 2 :(得分:1)
对于可以为空的字段,可以使用包装器原始类型java。例如,在您的“房间表”中使用Integer实例int。 像wrapper primitive type java中那样,可以为null,但原始类型类不能为null。并为使用notNull = true的原始字段生成此SQL代码,但在Generayion中使用Integer时,则使用notNull = false
答案 3 :(得分:0)
如果您有多个要标记为唯一的项目&根据您要在db中插入的内容,您可以使用复合主键。
对于Not null,Room提供了“@NonNull”注释,该注释添加在不能为null的字段上。
在下面提到的例如。卷号在每个类中是唯一的,但是2个不同的类可以具有相同的卷号。所以我们可以使用class& rollNumber作为复合主键&在db中唯一地插入。 示例:
@Entity(primaryKeys = {"rollNumber", "class"})
class Student {
@NonNull
private int rollNumber;
private String firstName;
private String lastName;
private int class;
}