美文网首页Android数据库高手秘籍
Android数据库高手秘籍(四)LitePal建立表关联

Android数据库高手秘籍(四)LitePal建立表关联

作者: as_pixar | 来源:发表于2020-01-13 15:16 被阅读0次

    咱们学会了使用LitePal来创建表和升级表的方式,今天探究一下LitePal来建立表与表之间的关联关系。还没有看过前一篇文章的朋友建议先去参考Android数据库高手秘籍(三)LitePal升级表

    LitePal的项目地址是:https://github.com/LitePalFramework/LitePal

    logo.png

    关联关系基础知识

    多个类之间相互关联引用,共同完成某项功能。数据库当中,多个表之间可以相互关联吗?当然可以!我们就先来学习一下数据库表关联的基础知识。

    表与表之间的关联关系一共有三种类型,一对一、多对一、和多对多,下面我们分别对这三种类型展开进行讨论。

    一对一

    表示两个表中的数据必须是一 一对应的关系。这种场景不是很常见,咱们通过例子来直观地体会一下。

    一个男人只能娶一个媳妇,我们创建一个man表,里面主要记录了男人的 姓名 年龄 身高,接着创建一个woman表,里面主要记录了 姓名 年龄 身高 。那么很显然,man表和woman表就是一对一的关系了,因为一个男人只能娶一个女人,一个女人也只能属于一个男人。它们之间的对应关系大概如下图描述的一样:

    image.png

    可以看到,Man1对应了Woman2,Man2对应了Woman3,Man3对应了Woman1,不管怎么样,它们都是一对一的关系。在编程语言中该怎么体现出来呢?

    只需要在Man类中持有一个Woman类的引用,然后在Woman类中也持有一个Man类的引用,这样它们之间自然就是一对一的关系了。对象之间的一对一关系非常简单易懂。如何在数据库表中建立这样的一对一关系了。

    由于数据库并不像面向对象的语言一样支持相互引用,如果想让两张表之间建立一对一的关系,一般就只能通过外键的方式来实现了。因此,一对一关系的表结构就可以这样设计:

    image.png

    请注意,woman表中有一个man_id列,这是一个外键列,里面应该存放一个具体的男人id,这样一条woman就能对应一条man,也就实现一对一的关系了,如下图所示:


    image.png

    由此我们就能够看出,id为1的woman对应着id为3的man,id为2的woman对应着id为1的man,id为3的woman对应着id为2的man。需要注意的是,一对一的关系并没有强制要求外键必须加在哪一张表上,你可以在woman表中加一个man_id作为外键,也可以在man表中加一个woman_id作为外键,不管使用哪一种,都可以表示出它们是一对一的关联关系。

    多对一

    表示一张表中的数据可以对应另一张表中的多条数据。这种场景比起一对一关系就要常见太多了,比如说现在我们的数据库中有一个news表,还有一个comment表,它们两个之间就是典型的多对一关系,一条新闻可以有很多条评论,但是一条评论只能是属于一条新闻的。它们的关系如下图所示:

    image.png

    而这种多对一的关系在编程语言中是非常容易体现出来的,比如Java中就有专门集合类,如List、Set等,使用它们的话就能轻松简单地在对象之间建立多对一的关系,如何在表中建立多对一关系。前面我们已经学会了一对一关系的建立方法,而多对一也是类似的。数据库表中多对一的关系仍然是通过外键来建立的,只不过一对一的时候外键加在哪一张表上都可以,但多对一的时候外键必须要加在多方的表中。因此,多对一关系的表结构就可以这样设计:

    image.png

    comment表中有一个news_id列,这是一个外键列,里面存放一个新闻id,并且允许多条comment都存放同一个新闻id,这样一条评论就只能对应一条新闻,但一条新闻却可以有多条评论,也就实现多对一的关系了,如下图所示:

    image.png

    由此我们就可以看出,id为1、2的两条评论是属于第一条新闻的,而id为3、4的两条评论是属于第二条新闻的。

    多对多

    表示两张关联表中的数据都可以对应另一张表中的多条数据。举个例子,我们都知道新闻网站是会将新闻进行分类,用户就可以选择自己喜欢的那一类新闻进行浏览,比如说网易新闻中就会有头条、科技、娱乐、手机等等种类。每个种类下面当然都会有许多条新闻,而一条新闻也可能是属于多个种类的,比如iPhone6发布的新闻既可以属于手机种类,也可以属于科技种类,甚至还可以上头条。因此,新闻和种类之间就是一种多对多的关系,如下图所示:

    image.png

    可以看到,News1是属于Category1的,而News2和News3都是既属于Category1也属于Category2,如此复杂的关联关系该如何表示呢?在面向对象的编程语言中一切都是那么的简单,只需要在News类中使用集合类声明拥有多个Category,然后在Category类中也使用集合类声明拥有多个News就可以了,我们稍后就会看到。而难点仍然是留在了数据库上,两张表之间如何建立多对多的关联关系呢,还是用外键吗?肯定不行了,多对多的情况只能是借助中间表来完成了。也就是说,我们需要多建立一张表,就是为了存放news表和category表之间的关联关系的,如下图所示:

    image.png

    注意这里我们建立一张名为category_news的中间表,中间表的命名并没有什么强制性的约束,但一个良好的命名规范可以让你一眼就明白这张表是用来做什么的。中间表里面只有两列,而且也只需要有两列,分别是news表的外键和category表的外键,在这里存放新闻和分类相应的id,就可以让它们之间建立关联关系了,如下图所示:

    image.png

    第一条新闻是属于第一个分类的,而第二条新闻,则既属于第一个种类,也属于第二个种类。反过来也可以这样看,第一个分类下面有第一、第二、第三这三条新闻,而第二个种类下面只有第二、这一条新闻。不管怎么看,多对多的关系都是成立的。

    三种关联关系都讲完了,那我们来简单总结。虽说上面介绍花了很大的篇幅讲解数据库的表关联知识,但其实最后的结论是非常简单的,大家可以当成口诀一样背下来。即一对一关联的实现方式是用外键,多对一关联的实现方式也是用外键,多对多关联的实现方式是用中间表。记下了这个口诀,在很多数据库设计的时候,你都可以发挥得更加游刃有余。

    使用LitePal建立表关联

    LitePal自动建立表关联又是一个非常不错的选择,我们不需要关心什么外键、中间表等实现的细节,只需要在对象中声明好它们相互之间的引用关系,LitePal就会自动在数据库表之间建立好相应的关联关系了,下面我们来尝试喽。

    首先确定一下一共涉及到了哪些实体类,News和Comment,这两个类我们在前两篇文章中就已经建好了,然后还需要有Man和Woman这两个类,新建Man类,代码如下所示:

    public class Man {
    
        private String name;
        private int age;
        private int height;
        // 自动生成get、set方法
    }
    

    接着新建Woman类,代码如下所示:

    public class Woman {
    
        private String name;
        private int age;
        private int height;
        // 自动生成get、set方法
    }
    

    现在四个类都已经建好了,但目前它们都还是各自独立的,互相之间没有任何联系,那么我们现在就开始用极为简单易懂的方式来给它们建立关联吧。首先,Man和Woman是一对一的关系,那就可以在Woman类中添加如下引用:

    public class Woman {
         ...
        private Man man;
        // 自动生成get、set方法
    }
    

    Man类中可以得到一个对应的Woman的实例,那么它们之间就是一对一关系了。

    接着Comment和News是多对一的关系,因此News中应该包含多个Comment,而Comment中应该只有一个News,所以就可以这样写:

    public class News {
        ...
    
        private List<Comment> commentList = new ArrayList<Comment>();
        // 自动生成get、set方法
    }
    

    先使用一个泛型为Comment的List集合来表示News中包含多个Comment,然后修改Comment类的代码,如下所示:

    public class Comment {
        ...
        private News news;
        
        // 自动生成get、set方法 
    }
    

    在Comment类中声明了一个News的实例,这样就清楚地表示出了News中可以包含多个Comment,而Comment中只能有一个News,也就是多对一的关系了。

    最后News和Category是多对多的关系,相信聪明的你一定已经知道该怎么写了。News中可以包含多个Category,所以仍然应该使用List集合来表示:

    public class News {
        ...
        
        private List<Comment> commentList = new ArrayList<Comment>();
        
        private List<Category> categoryList = new ArrayList<Category>();
        
        // 自动生成get、set方法
    }
    

    而Category中也可以包含多个News,因此Category类也应该使用相同的写法,如下所示:

    public class Category {
        ...
        private List<News> newsList = new ArrayList<News>();
        
        // 自动生成get、set方法
    }
    

    这样就清楚地表达出它们之间是多对多的关联了。

    关联关系都声明好了之后,我们只需要将所有的实体类都添加到映射列表当中,并将数据库版本号加1就可以了。修改litepal.xml的代码,如下所示:

    <?xml version="1.0" encoding="utf-8"?>
    <litepal>
        <dbname value="demo" />
    
        <version value="7" />
    
        <list>
            <mapping class="com.example.sqlitefeel.bean.News" />
            <mapping class="com.example.sqlitefeel.bean.Comment" />
            <mapping class="com.example.sqlitefeel.bean.Man" />
            <mapping class="com.example.sqlitefeel.bean.Woman" />
            <mapping class="com.example.sqlitefeel.bean.Category" />
        </list>
    </litepal>
    

    现在只需要任意操作一下数据库,表之间的关联关系就将会自动建立,比如说调用一下Connector.getDatabase()方法。

    下面我们来验证一下吧,输入.table命令查看一下当前数据库中的表,如下所示:
    sqlite> .tables
    android_metadata category_news man table_schema
    category comment news woman

    哈哈,news、comment、category、man、woman这几张表全都有了,除此之外还有一张category_news中间表。那我们要来一 一检查一下了,先查看一下man表的结构吧,如下所示:

    sqlite> pragma table_info(man);
    0|id|integer|0||1
    1|age|integer|0||0
    2|height|integer|0||0
    3|name|text|0||0
    4|woman_id|integer|0||0
    

    可以看到,多了一个woman_id列,说明man表和woman表之间的一对一关系已经建立好了。

    然后再检查一下woman表的结构,如下所示:

    sqlite> pragma table_info(woman);
    0|id|integer|0||1
    1|age|integer|0||0
    2|height|integer|0||0
    3|name|text|0||0
    4|man_id|integer|0||0
    

    可以看到,多了一个man_id列,说明man表和woman表之间的一对一关系已经建立好了。

    然后再检查一下comment表的结构,如下所示:

    sqlite> pragma table_info(comment);
    0|id|integer|0||1
    1|content|text|0||0
    2|publishdate|integer|0||0
    3|news_id|integer|0||0
    

    哦耶,comment表中也有一个news_id的列,那么comment表和news表之间的多对一关系也已经建立好了。
    瞅一眼news表

    sqlite> pragma table_info(news);
    0|id|integer|0||1
    1|content|text|0||0
    2|publishdate|integer|0||0
    3|title|text|0||0
    

    最后检查一下category_news这张中间表的结构,如下所示:

    sqlite> pragma table_info(category_news);
    0|category_id|integer|0||0
    1|news_id|integer|0||0
    

    一共只有两列,一列是news_id,一列是category_id,分别对应着两张表的外键,这样news表和category表的多对多关系也建立好了。

    创建表、升级表、表关联,LitePal在数据库表管理方面给我们带来的巨大便利,咱么体会了它的魅力所在了。LitePal进行表管理的知识全部学完了,下一篇文章,我将会讲解如何使用LitePal进行CRUD的操作。感兴趣的朋友请继续阅读 Android数据库高手秘籍(五)LitePal的存储操作 https://www.jianshu.com/p/de87516a1d72

    相关文章

      网友评论

        本文标题:Android数据库高手秘籍(四)LitePal建立表关联

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