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

本文详细介绍了数据库中一对一、多对一和多对多三种表关联关系,并通过例子展示了如何在编程中体现这些关系。在 LitePal 中,通过在实体类中声明引用关系,可以自动建立表之间的关联,简化了数据库表管理。最后,通过示例验证了 LitePal 在实际操作中的效果。
摘要由CSDN通过智能技术生成

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

df0c52c1a771

logo.png

关联关系基础知识

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

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

一对一

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

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

df0c52c1a771

image.png

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

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

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

df0c52c1a771

image.png

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

df0c52c1a771

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表,它们两个之间就是典型的多对一关系,一条新闻可以有很多条评论,但是一条评论只能是属于一条新闻的。它们的关系如下图所示:

df0c52c1a771

image.png

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

df0c52c1a771

image.png

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

df0c52c1a771

image.png

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

多对多

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

df0c52c1a771

image.png

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

df0c52c1a771

image.png

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

df0c52c1a771

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 commentList = new ArrayList();

// 自动生成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 commentList = new ArrayList();

private List categoryList = new ArrayList();

// 自动生成get、set方法

}

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

public class Category {

...

private List newsList = new ArrayList();

// 自动生成get、set方法

}

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

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

现在只需要任意操作一下数据库,表之间的关联关系就将会自动建立,比如说调用一下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。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值