房间持久性库没有“NOT NULL”和“UNIQUE”约束

时间:2017-09-04 05:23:36

标签: android sqlite android-room

在玩房间持久性库时,我发现没有方法可以使用NOT NULL和UNIQUE约束设置数据类字段。 SQLite是否支持这些约束。迁移使用这些约束的旧数据库不是一个问题吗?任何人都可以就此问题提出建议吗?

4 个答案:

答案 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时,则使用no​​tNull = 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;
        }