码迷,mamicode.com
首页 > 数据库 > 详细

【mysql】逗号分割字段的行列转换

时间:2017-12-29 16:56:45      阅读:290      评论:0      收藏:0      [点我收藏+]

标签:根据   分隔符   连续   join   ace   inno   engine   tab   inf   

  由于很多业务表因为历史原因或者性能原因,都使用了违反第一范式的设计模式,即同一个列中存储了多个属性值。这种模式下,应用常常需要将这个列依据分隔符进行分割,并得到列转行的结果:

建表语句:

 1 DROP table if EXISTS tbl_name;
 2 CREATE TABLE tbl_name(
 3     id       int(11)    not null auto_increment,
 4     userName varchar(100)    not null,
 5 PRIMARY KEY(id)
 6 )
 7 ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8;
 8 
 9 insert into tbl_name values (1,a,aa,aaa);
10 insert into tbl_name values (2,b,bb);
11 insert into tbl_name values (3,c,cc);

如下图:

技术分享图片

sql语句:

1 SELECT a.id,SUBSTRING_INDEX(SUBSTRING_INDEX(a.userName,,,b.help_topic_id+1),,,-1) as name  
2 from tbl_name a left join mysql.help_topic b 
3 on b.help_topic_id < (LENGTH(a.userName)-LENGTH(REPLACE(a.userName,,,‘‘))+1) 
4 ORDER BY a.id;

执行结果:

技术分享图片

分析如下:

LENGTH(a.userName)-LENGTH(REPLACE(a.userName,,,‘‘))+1

表示了按逗号分割后,获得行转成列的数量,以下简称n;

根据id进行循环

{
  判断:i 是否 <= n
    {
      获取最靠近第 i 个逗号之前的数据, 即 SUBSTRING_INDEX(SUBSTRING_INDEX(a.userName,,,b.help_topic_id+1),,,-1)
      i = i +1 
    }
  id = id +1 
}

总结:

这种方法的缺点在于,我们需要一个拥有连续数列的独立表(这里是incre_table)。并且连续数列的最大值一定要大于符合分割的值的个数。当然,mysql内部也有现成的连续数列表可用。如mysql.help_topic: help_topic_id 共有504个数值,一般能满足于大部分需求了。

【mysql】逗号分割字段的行列转换

标签:根据   分隔符   连续   join   ace   inno   engine   tab   inf   

原文地址:https://www.cnblogs.com/hongzm/p/8145149.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!