加入收藏 | 设为首页 | 会员中心 | 我要投稿 银川站长网 (https://www.0951zz.com/)- 云通信、基础存储、云上网络、机器学习、视觉智能!
当前位置: 首页 > 站长学院 > MsSql教程 > 正文

SQL语句中JOIN的用法是怎样 可以做什么

发布时间:2023-05-12 11:17:44 所属栏目:MsSql教程 来源:
导读:写SQL最高境界:SELECT * FROM 表名。当然这是一句自嘲。探究一下SQL语句中JOIN的用法,直到经历这个场景,变得想验证一下究竟。一、场景把关系型数据库A中表TEST_TB01和TEST_TB02迁移到大数据平台M(MaxCompute大数

写SQL最高境界:SELECT * FROM 表名。当然这是一句自嘲。探究一下SQL语句中JOIN的用法,直到经历这个场景,变得想验证一下究竟。

一、场景

把关系型数据库A中表TEST_TB01和TEST_TB02迁移到大数据平台M(MaxCompute大数据平台)。TEST_TB01单表1000万条记录,TEST_TB02单表80万条记录。

在关系型数据库中,TEST_TB01和TEST_TB02中有主键约束。在产生新增业务数据时,不会存在重复数据插入。但是,当数据迁移到大数据平台后,由于在大数据平台中无主键约束功能。在产生新增业务数据时,TEST_TB01和TEST_TB02均均插入了重复数据。

在一个计算任务中,TEST_TB01和TEST_TB02根据某个字段JOIN连接,计算出了一份结果数据,数据推送到使用方的关系型数据库C。直接导致了C数据库的对应表的表空间撑爆,监控预警。

原因:TEST_TB01和TEST_TB02有重复数据,使用JOIN连接后,生成了10亿+条数据,共计200G+数据,直接推送到C数据库。

那次考虑不周,瞬间懵了,感觉SQL语句中的JOIN变得陌生极了。于是想探究一下以作记录。

二、建表

TEST_TB01建表语句:

create table TEST_TB01

(

sensor_id BIGINT,

part_id BIGINT

)

COMMENT '数据表一';

TEST_TB02建表语句:

create table TEST_TB02

(

part_id BIGINT,

elem_id BIGINT

)

COMMENT '数据表二';

三、SQL语句中使用JOIN无重复数据情况

在SQL语句中使用JOIN无重复数据情况,即在TEST_TB01和TEST_TB02表中均无重复数据情况。分别使用JOIN、INNER JOIN、LEFT JOIN、LEFT OUTER JOIN、RIGHT JOIN、FULL JOIN验证。

在TEST_TB01插入数据:

insert into TEST_TB01 (sensor_id,part_id) values(2101,9911);

insert into TEST_TB01 (sensor_id,part_id) values(2102,9912);

insert into TEST_TB01 (sensor_id,part_id) values(2103,9913);

insert into TEST_TB01 (sensor_id,part_id) values(2104,9914);

insert into TEST_TB01 (sensor_id,part_id) values(2105,9915);

在TEST_TB02插入数据:

insert into TEST_TB02 (part_id,elem_id) values(9911,8901);

insert into TEST_TB02 (part_id,elem_id) values(9912,8902);

insert into TEST_TB02 (part_id,elem_id) values(9913,8903);

insert into TEST_TB02 (part_id,elem_id) values(9916,8906);

1.在SQL中使用JOIN

TEST_TB01和TEST_TB02根据part_id使用JOIN连接,只返回两个表(TEST_TB01和TEST_TB02)中连接字段相等的记录。

SQL语句:

SELECT

*

FROM

TEST_TB01 aa

JOIN TEST_TB02 bb

ON aa.part_id = bb.part_id

ORDER BY aa.sensor_id ASC;

2.在SQL中使用INNER JOIN

TEST_TB01和TEST_TB02根据part_id使用INNER JOIN连接,只返回两个表(TEST_TB01和TEST_TB02)中连接字段相等的记录。INNER JOIN和JOIN效果等价。

SQL语句:

SELECT

*

FROM

TEST_TB01 aa

INNER JOIN TEST_TB02 bb

ON aa.part_id = bb.part_id

ORDER BY aa.sensor_id ASC;

3.在SQL中使用LEFT JOIN

TEST_TB01和TEST_TB02根据part_id使用LEFT JOIN连接,左连接,返回左表(TEST_TB01)中所有的记录以及右表(TEST_TB02)中连接字段相等的记录。

SQL语句:

SELECT

*

FROM

TEST_TB01 aa

LEFT JOIN TEST_TB02 bb

ON aa.part_id = bb.part_id

ORDER BY aa.sensor_id ASC;

4.在SQL中使用LEFT OUTER JOIN

TEST_TB01和TEST_TB02根据part_id使用LEFT OUTER JOIN连接,左外连接,返回左表(TEST_TB01)中所有的记录以及右表(TEST_TB02)中连接字段相等的记录。LEFT OUTER JOIN

和LEFT JOIN等价。

SQL语句:

SELECT

*

FROM

TEST_TB01 aa

LEFT OUTER JOIN TEST_TB02 bb

ON aa.part_id = bb.part_id

ORDER BY aa.sensor_id ASC;

5.在SQL中使用RIGHT JOIN

TEST_TB01和TEST_TB02根据part_id使用RIGHT JOIN连接,右连接,返回右表(TEST_TB02)中所有的记录以及左表(TEST_TB01)中连接字段相等的记录

SQL语句:

SELECT

*

FROM

TEST_TB01 aa

RIGHT JOIN TEST_TB02 bb

ON aa.part_id = bb.part_id

ORDER BY aa.sensor_id ASC;

6.在SQL中使用FULL JOIN

TEST_TB01和TEST_TB02根据part_id使用FULL JOIN连接,外连接,返回两个表中的行:LEFT JOIN + RIGHT JOIN所有行记录。

SQL语句:

SELECT

*

FROM

TEST_TB01 aa

FULL JOIN TEST_TB02 bb

ON aa.part_id = bb.part_id

ORDER BY aa.sensor_id ASC;

四、SQL语句中使用JOIN有重复数据情况

在SQL语句中使用JOIN有重复数据情况,即在TEST_TB01和TEST_TB02表中均有重复数据情况。分别使用JOIN、INNER JOIN、LEFT JOIN、LEFT OUTER JOIN、RIGHT JOIN、FULL JOIN验证。

在TEST_TB01插入数据:

insert into TEST_TB01 (sensor_id,part_id) values(2101,9911);

insert into TEST_TB01 (sensor_id,part_id) values(2102,9912);

insert into TEST_TB01 (sensor_id,part_id) values(2103,9913);

insert into TEST_TB01 (sensor_id,part_id) values(2104,9914);

insert into TEST_TB01 (sensor_id,part_id) values(2105,9915);

--造重复数据

insert into TEST_TB01 (sensor_id,part_id) values(2102,9912);

insert into TEST_TB01 (sensor_id,part_id) values(2103,9913);

在TEST_TB02插入数据:

insert into TEST_TB02 (part_id,elem_id) values(9911,8901);

insert into TEST_TB02 (part_id,elem_id) values(9912,8902);

insert into TEST_TB02 (part_id,elem_id) values(9913,8903);

insert into TEST_TB02 (part_id,elem_id) values(9916,8906);

--造重复数据

insert into TEST_TB02 (part_id,elem_id) values(9912,8902);

insert into TEST_TB02 (part_id,elem_id) values(9913,8903);

1.在SQL中使用JOIN

TEST_TB01和TEST_TB02根据part_id使用JOIN连接,只返回两个表(TEST_TB01和TEST_TB02)中连接字段相等的记录。

SQL语句:

SELECT

*

FROM

TEST_TB01 aa

JOIN TEST_TB02 bb

ON aa.part_id = bb.part_id

ORDER BY aa.sensor_id ASC;

2.在SQL中使用INNER JOIN

TEST_TB01和TEST_TB02根据part_id使用INNER JOIN连接,只返回两个表(TEST_TB01和TEST_TB02)中连接字段相等的记录。INNER JOIN和JOIN效果等价。

SQL语句:

SELECT

*

FROM

TEST_TB01 aa

INNER JOIN TEST_TB02 bb

ON aa.part_id = bb.part_id

ORDER BY aa.sensor_id ASC;

3.在SQL中使用LEFT JOIN

TEST_TB01和TEST_TB02根据part_id使用LEFT JOIN连接,左连接,返回左表(TEST_TB01)中所有的记录以及右表(TEST_TB02)中连接字段相等的记录。

SQL语句:

SELECT

*

FROM

TEST_TB01 aa

LEFT JOIN TEST_TB02 bb

ON aa.part_id = bb.part_id

ORDER BY aa.sensor_id ASC;

4.在SQL中使用LEFT OUTER JOIN

TEST_TB01和TEST_TB02根据part_id使用LEFT OUTER JOIN连接,左外连接,返回左表(TEST_TB01)中所有的记录以及右表(TEST_TB02)中连接字段相等的记录。LEFT OUTER JOIN

和LEFT JOIN等价。

SQL语句:

SELECT

*

FROM

TEST_TB01 aa

LEFT OUTER JOIN TEST_TB02 bb

ON aa.part_id = bb.part_id

ORDER BY aa.sensor_id ASC;

5.在SQL中使用RIGHT JOIN

TEST_TB01和TEST_TB02根据part_id使用RIGHT JOIN连接,右连接,返回右表(TEST_TB02)中所有的记录以及左表(TEST_TB01)中连接字段相等的记录

SQL语句:

SELECT

*

FROM

TEST_TB01 aa

RIGHT JOIN TEST_TB02 bb

ON aa.part_id = bb.part_id

ORDER BY aa.sensor_id ASC;

6.在SQL中使用FULL JOIN

TEST_TB01和TEST_TB02根据part_id使用FULL JOIN连接,外连接,返回两个表中的行:LEFT JOIN + RIGHT JOIN所有行记录。

SQL语句:

SELECT

*

FROM

TEST_TB01 aa

FULL JOIN TEST_TB02 bb

ON aa.part_id = bb.part_id

ORDER BY aa.sensor_id ASC;

五、SQL中使用JOIN有重复与无重复数据区别

在SQL语句中使用JOIN有重复数据情况,使用JOIN连接,符合连接字段相等的记录的结果集是笛卡尔积,第一个表的行数乘以第二个表的行数。

六、解决方式

1.先去重再使用JOIN连接

根据业务规则先对TEST_TB01和TEST_TB02分别去重再使用JOIN连接。

2.先使用JOIN连接再去重

根据业务规则先对TEST_TB01和TEST_TB02使用JOIN连接生成结果集,再对结果集去重。

3.建议

在生产环境特别是数据量大场景,推荐使用第一种方式,先逐个表去重再使用JOIN连接。

七、关系型数据库验证表结构

本例是在DataWorks环境(即MaxCompute大数据平台)下验证,即在关系型数据库验证除表结构差异,其它均相同。

在ORACLE数据库建表语句:

create table TEST_TB01

(

sensor_id NUMBER(16),

part_id NUMBER(16)

);

create table TEST_TB02

(

part_id NUMBER(16),

elem_id NUMBER(16)

);

在MySQL数据库建表语句:

CREATE TABLE TEST_TB01

(

sensor_id BIGINT,

part_id BIGINT

);

CREATE TABLE TEST_TB02

(

part_id BIGINT,

elem_id BIGINT

);

上述内容具有一定的借鉴价值,感兴趣的朋友可以参考,希望能对大家有帮助。

(编辑:银川站长网)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章