美文网首页Android开发Android开发经验谈Android知识
Android官方数据库架构Room使用解析一

Android官方数据库架构Room使用解析一

作者: dreamruner | 来源:发表于2018-02-05 21:23 被阅读581次

    使用Room Entity定义数据

    使用Room持久化库时,可将相关字段的集合定义为实体.对于每个实体,在关联的数据库对象中创建一个表来保存这些字段。

    注意 :使用实体在你的app 内,添加架构组件依赖到你appbuild.gradle 文件中.

    默认的,Room 为实体中定义的每一个字段创建一个列,如果实体中包含你不想持久化的字段,你可以使用@Ignore注解它们,你必须通过Database 类中的实体数组引用实体类。

    以下的代码片段显示了怎么去定义一个实体:

    @Entity
    class User {
        @PrimaryKey
        public int id;
    
        public String firstName;
        public String lastName;
    
        @Ignore
        Bitmap picture;
    }
    

    为了持久化一个字段,Room必须有访问它的权限.你可以用public修饰一个字段,或者为它提供setget方法.如果你使用getset方法,请记住这是基于Roomjavabean的约定。

    注意 :实体可以有一个空构造函数(如果相应的DAO 类可以访问每个持久化字段)或者一个构造函数的参数包含与实体类的字段匹配的类型和名称.Room也能使用全部或部分构造函数,例如一个构造函数只接收一些字段.

    使用主键

    每一个实体必须至少定义一个字段作为主键.即使该实体只有一个字段,你仍然需要使用@PrimaryKey注解这个字段.并且,如果你想Room为实体自动分配ID,你可以设置@PrimaryKey's autoGenerate 属性.如果实体具有复合主键,你可以使用 @EntityprimaryKeys 属性,如下面的代码片段所示:

    @Entity(primaryKeys = {"firstName", "lastName"})
    class User {
        public String firstName;
        public String lastName;
    
        @Ignore
        Bitmap picture;
    }
    

    默认的,Room 使用类名作为数据库的表名.如果你想给表设置不同的名字,给@Entity 注解设置tableName 属性,如以下代码片段所示:

    @Entity(tableName = "users")
    class User {
        ...
    }
    

    警告 :表名在SQLite 数据库中是大小写敏感的.

    tableName 属性类似,Room使用字段名字作为数据库表的列名.如果你想给一列设置不同的名字,给这个字段添加

    @ColumnInfo 注解,如下代码片段所示:

    @Entity(tableName = "users")
    class User {
        @PrimaryKey
        public int id;
    
        @ColumnInfo(name = "first_name")
        public String firstName;
    
        @ColumnInfo(name = "last_name")
        public String lastName;
    
        @Ignore
        Bitmap picture;
    }
    

    注释索引和唯一性

    取决于你如何访问你的数据,你可能想给数据库的某些字段添加索引来加快你的查询速度.为实体添加索引,可以在@Entity注解中包含 indices 属性,列出要包含在索引或复合索引中的列的名称,以下代码片段演示了这个注释过程:

    @Entity(indices = {@Index("name"),
            @Index(value = {"last_name", "address"})})
    class User {
        @PrimaryKey
        public int id;
    
        public String firstName;
        public String address;
    
        @ColumnInfo(name = "last_name")
        public String lastName;
    
        @Ignore
        Bitmap picture;
    }
    

    有时,数据库中的某些字段或字段组必须是唯一的.您可以通过将@Index注解的唯一属性设置为true来强制执行此unique 属性,下面的代码示例防止表中有两行包含firstNamelastName列的相同值集合:

    @Entity(indices = {@Index(value = {"first_name", "last_name"},
            unique = true)})
    class User {
        @PrimaryKey
        public int id;
    
        @ColumnInfo(name = "first_name")
        public String firstName;
    
        @ColumnInfo(name = "last_name")
        public String lastName;
    
        @Ignore
        Bitmap picture;
    }
    

    定义对象之间的关系

    因为SQLite 是一个关系数据库,所以你可以指定对象之间的关系。即使大多数对象关系映射库允许实体对象相互引用,Room明确禁止这一点.了解这个决定背后的技术推理,查看 理解Room为什么不允许对象引用.

    即使您不能使用直接关系,Room 仍允许您定义实体之间的外键约束。

    例如,如果有另一个名为Book 的实体,则可以使用@ForeignKey 注释来定义它与User 实体的关系,如以下代码片段所示:

    @Entity(foreignKeys = @ForeignKey(entity = User.class,
                                      parentColumns = "id",
                                      childColumns = "user_id"))
    class Book {
        @PrimaryKey
        public int bookId;
    
        public String title;
    
        @ColumnInfo(name = "user_id")
        public int userId;
    }
    

    外键非常强大,因为它们允许您指定在引用实体更新时发生的情况.例如,如果通过在@ForeignKey注解中包含onDelete = CASCADE来删除用户的相应实例,则可以让SQLite删除用户的所有书籍。

    注意: SQLite@Insert(onConflict = REPLACE)作为一组REMOVEREPLACE操作处理,而不是单个UPDATE操作,这种替换冲突值的方法可能会影响您的外键约束。有关更多详细信息,请参阅SQLite documentation ON_CONFLICT子句的SQLite文档。

    创建嵌套的对象

    有时候,您希望在数据库逻辑中将实体或简单的旧Java 对象(POJO)表示为一个有凝聚力的整体,即使对象包含几个字段.在这些情况下,您可以使用@Embedded注解来表示要分解到表中的子字段的对象,然后,您可以像查看其他单个列一样查询嵌入字段。

    例如,我们的User 类可以包含一个Address 类型的字段,它表示一个名为streetcitystatepostCode 的字段的组合。在表格中分别存储组成的列,请在User 类中包含一个用@Embedded 注解的Address 字段,如下面的代码片段所示:

    class Address {
        public String street;
        public String state;
        public String city;
    
        @ColumnInfo(name = "post_code")
        public int postCode;
    }
    
    @Entity
    class User {
        @PrimaryKey
        public int id;
    
        public String firstName;
    
        @Embedded
        public Address address;
    }
    

    表示User对象的表格包含具有以下名称的列:idfirstNamestreetstatecitypost_code

    注意 : 嵌入字段也可以包含其他嵌入字段。

    如果实体具有多个相同类型的嵌入字段,则可以通过设置prefix 属性来保持每个列的唯一性。然后,Room 会将提供的值添加到嵌入对象中每个列名的开头。

    相关文章

      网友评论

        本文标题:Android官方数据库架构Room使用解析一

        本文链接:https://www.haomeiwen.com/subject/leaizxtx.html