点击上方蓝字关注我

a469aebd34cfe735b45f3024025d40db.png

1. 创建测试表及数据

-- 创建一张tb_stu表,
CREATE TABLE tb_user(
id INT PRIMARY KEY AUTO_INCREMENT,
NAME VARCHAR(10) COMMENT '人名',
c_no VARCHAR(64) COMMENT '持剑ID,以逗号分隔'
);
INSERT INTO tb_user(NAME,c_no) VALUES('蘧伯玉','1,3,5,7');
INSERT INTO tb_user(NAME,c_no) VALUES('高渐离','1,2,4,8,5');
INSERT INTO tb_user(NAME,c_no) VALUES('樗里疾','2,9');
INSERT INTO tb_user(NAME,c_no) VALUES('澹台灭明','1,2');
INSERT INTO tb_user(NAME,c_no) VALUES('钟子期','1,2,6,8,7,3,5');
INSERT INTO tb_user(NAME,c_no) VALUES('柳下惠','2,4,3,5');
INSERT INTO tb_user(NAME,c_no) VALUES('百里奚','1,9');
INSERT INTO tb_user(NAME,c_no) VALUES('阚止','1,6,7');
INSERT INTO tb_user(NAME,c_no) VALUES('霍去病','1,8,5');
INSERT INTO tb_user(NAME,c_no) VALUES('慕容白曜','1,2,3,4,5,7');
INSERT INTO tb_user(NAME,c_no) VALUES('鱼幼薇','7,8,9');
INSERT INTO tb_user(NAME,c_no) VALUES('宋玉','6,5');
-- 创建一张剑名
create table tb_sword(
id int primary key AUTO_INCREMENT,
c_name varchar(4)) comment '剑名';


insert into  tb_sword(c_name)values('轩辕');
insert into  tb_sword(c_name)values('湛卢');
insert into  tb_sword(c_name)values('赤霄');
insert into  tb_sword(c_name)values('太阿');
insert into  tb_sword(c_name)values('七星龙渊');
insert into  tb_sword(c_name)values('干将');
insert into  tb_sword(c_name)values('莫邪');
insert into  tb_sword(c_name)values('鱼肠');
insert into  tb_sword(c_name)values('纯钧');

两张表内容如下: 

tb_user

5a6c552b9065e7f3d3f116b6a82f58c5.png

tb_sword

619c61e9dff4ab1bfb7f186de1a6253c.png

2.  数据拆分及合并

需求: 使用一条SQL获得tb_user表中每个人持有的剑名(剑名用“|”分隔),即得到如下结果

a45f7aef390cd04f50b67c31275fad3d.png

拆解需求:  

1) 先将tb_user表中的c_no按逗号拆分

2)将拆分后c_no中的各个id与tb_sword中的id关联,获取剑名

3)  最后将每一个user对应的剑名合并成一个字段

分段SQL如下:

步骤1:

每一个user的c_no按逗号拆分为对应的c_id,此方法需借助于mysql.help_topic表

SELECT a.id,a.name,a.c_no,
SUBSTRING_INDEX( SUBSTRING_INDEX( a.c_no, ',', b.help_topic_id + 1 ), ',',- 1 ) c_id
FROM tb_user a
JOIN mysql.help_topic b ON b.help_topic_id < ( LENGTH( a.c_no ) - LENGTH( REPLACE ( a.c_no, ',', '' ) ) + 1 )
ORDER BY a.id

结果如下:

948806232fef348b97b27444ecb49c44.png

步骤2:关联获取每个id对应的剑名

SELECT a2.id,a2.name,a2.c_no,a2.c_id,b2.c_name 
FROM (SELECT a.id,a.name,a.c_no,
SUBSTRING_INDEX( SUBSTRING_INDEX( a.c_no, ',', b.help_topic_id + 1 ), ',',- 1 ) c_id
FROM tb_user a
JOIN mysql.help_topic b ON b.help_topic_id < ( LENGTH( a.c_no ) - LENGTH( REPLACE ( a.c_no, ',', '' ) ) + 1 )
ORDER BY a.id) a2,   -- a2表即步骤1中拆分的结果
tb_sword b2
WHERE a2.c_id =b2.id  -- 关联,相当于inner join(或者join)

结果如下

287fd8613e4e93176597e115c9de4eba.png

步骤3:

将每个人的剑名合并为1个字段显示,并用"|" 符合合并

SELECT a2.id,a2.name,a2.c_no,  
GROUP_CONCAT(b2.c_name SEPARATOR '|' ) sword_name  
-- SEPARATOR 指定分隔富,不加默认为逗号分隔
FROM (SELECT a.id,a.name,a.c_no,
SUBSTRING_INDEX( SUBSTRING_INDEX( a.c_no, ',', b.help_topic_id + 1 ), ',',- 1 ) c_id
FROM tb_user a
JOIN mysql.help_topic b ON b.help_topic_id < ( LENGTH( a.c_no ) - LENGTH( REPLACE ( a.c_no, ',', '' ) ) + 1 )
ORDER BY a.id) a2,tb_sword b2
WHERE a2.c_id =b2.id
GROUP BY a2.id

结果如下:

459c3547b040be772324f40dedba5a76.png

实现需求

55575a91a7986a5d3490d55144f5cc84.png

往期精彩回顾

1.  MySQL高可用之MHA集群部署

2.  mysql8.0新增用户及加密规则修改的那些事

3.  比hive快10倍的大数据查询利器-- presto

4.  监控利器出鞘:Prometheus+Grafana监控MySQL、Redis数据库

5.  PostgreSQL主从复制--物理复制

6.  MySQL传统点位复制在线转为GTID模式复制

7.  MySQL敏感数据加密及解密

8.  MySQL数据备份及还原(一)

9.  MySQL数据备份及还原(二)

fac620b342f90b3030e7db033afbf64f.png

扫码关注     

4556efe33b63f21d9fbe77335814aaae.png

33b250d60d693a3ced190793f3997a8a.png

d596a467fb3ea5976d18354be18177e9.png

Logo

为开发者提供学习成长、分享交流、生态实践、资源工具等服务,帮助开发者快速成长。

更多推荐