AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / dba / 问题 / 112408
Accepted
Tufan Chand
Tufan Chand
Asked: 2015-08-28 04:51:51 +0800 CST2015-08-28 04:51:51 +0800 CST 2015-08-28 04:51:51 +0800 CST

在 SQL Server 中加入逗号分隔值

  • 772

我有一张桌子[CourseMaster]喜欢

CourseId    CourseName
-----------------------
  01          ABC
  02          DEF
  03          GHI
  04          JKL
  05          MNO
  06          PQR
  07          STU

我还有一张[StudentMaster]学生详细信息表

ROLLNO  NAME    ADDRESS          Course
------------------------------------------------
12345   RAM     RAM ADDRESS      01,02,06                      
25695   HARI    HARI ADDRESS     02,06                         
89685   JEFF    JEFF ADDRESS     03,05,06,07                   
47896   DAISY   DAISY ADDRESS    03         

在这里,我想使用CourseName(Not CourseId) 获取学生详细信息。

如果 in 中的值Course不是逗号分隔的,那么query使用 join 获取详细信息将非常简单。

据我所知,我可以运行两个queries以获得我想要的相同结果,一个查询用于从[StudentMaster]前端获取学生的详细信息。另一个仅通过循环对应来获取CourseNamefrom 。[CourseMaster]CourseId

但事实上我只想要一个结果query而不是queries为这个小任务写两个。

我想这是 100% 可能的。我的预期结果将如下所示:

ROLLNO  NAME    ADDRESS         Course
-------------------------------------------
12345   RAM     RAM ADDRESS     ABC,DEF,PQR                   
25695   HARI    HARI ADDRESS    DEF,PQR                       
89685   JEFF    JEFF ADDRESS    GHI,MNO,PQR,STU               
47896   DAISY   DAISY ADDRESS   GHI                 

谢谢您,任何有价值的建议将不胜感激。

sql-server join
  • 4 4 个回答
  • 57257 Views

4 个回答

  • Voted
  1. Best Answer
    Aaron Bertrand
    2015-08-28T05:54:00+08:002015-08-28T05:54:00+08:00

    你真的应该有一个学生正在学习的课程的连接表,而不是将逗号分隔的值塞进一个单独的元组中。如果您认为这是由于这种次优设计而导致的最后一个问题,那么您将大吃一惊。你真的应该让这个项目的所有者阅读规范化 - 是的,改变你的模式很痛苦,但不断处理保持原样的限制也是如此。

    无论如何,话虽如此,您需要一个拆分功能。由于您的逗号分隔值是数字,因此您可以使用我的 XML 函数的变体;在这篇博文中还有其他几个可供选择。

    CREATE FUNCTION dbo.SplitStrings_XML
    (
       @List       VARCHAR(MAX),
       @Delimiter  CHAR(1) = ','
    )
    RETURNS TABLE
    WITH SCHEMABINDING
    AS
       RETURN 
       (  
          SELECT Item = y.i.value('(./text())[1]', 'varchar(8000)')
          FROM 
          ( 
            SELECT x = CONVERT(XML, '<i>' 
              + REPLACE(@List, @Delimiter, '</i><i>') 
              + '</i>').query('.')
          ) AS a CROSS APPLY x.nodes('i') AS y(i)
       );
    

    现在,您的查询是:

    ;WITH x AS 
    (
      SELECT s.ROLLNO, s.Name, s.Address, c.CourseId, c.CourseName
      FROM dbo.StudentMaster AS s
      CROSS APPLY dbo.SplitStrings_XML(s.Course, default) AS f
      INNER JOIN dbo.CourseMaster AS c
      ON f.item = c.CourseId
    )
    SELECT ROLLNO, Name, Address, STUFF((
      SELECT ',' + CourseName FROM x AS x2 
      WHERE x2.ROLLNO = x.ROLLNO
      ORDER BY CourseId FOR XML PATH, 
      TYPE).value(N'.[1]',N'varchar(max)'), 1, 1, '')
    FROM x
    GROUP BY ROLLNO, Name, Address;
    

    同样,这是一个复杂的解决方案,并且由于您的数据库结构较差,您必须执行的下一个查询将同样复杂和繁琐。几乎所有关于该主题的博客、文章或书籍都反对这种类型的设计是有原因的……

    在更现代的版本(SQL Server 2017 及更高版本)中,您可以组合STRING_AGG()并STRING_SPLIT()使其更简单:

    SELECT s.ROLLNO, s.NAME, s.ADDRESS, STRING_AGG(c.CourseName, ',')
      FROM dbo.StudentMaster AS s
      CROSS APPLY STRING_SPLIT(s.Course, ',') AS f
      INNER JOIN dbo.CourseMaster AS c
      ON f.value = c.CourseId
      GROUP BY s.ROLLNO, s.NAME, s.ADDRESS;
    
    • 9
  2. Mikael Eriksson
    2015-08-28T23:03:20+08:002015-08-28T23:03:20+08:00

    CourseMaster.CourseId与 Aaron Bertrand 在构建逗号分隔值时提供的解决方案相同,但在连接StudentMaster.Course.

    SQL小提琴

    MS SQL Server 2014 架构设置:

    create table dbo.CourseMaster
    (
      CourseId char(2),
      CourseName char(3)
    );
    
    create table dbo.StudentMaster
    (
      ROLLNO char(5),
      NAME varchar(10),
      ADDRESS varchar(20),
      Course varchar(100)
    );
    
    insert into dbo.CourseMaster values
    ('01', 'ABC'),
    ('02', 'DEF'),
    ('03', 'GHI'),
    ('04', 'JKL'),
    ('05', 'MNO'),
    ('06', 'PQR'),
    ('07', 'STU');
    
    insert into dbo.StudentMaster values
    ('12345', 'RAM',   'RAM ADDRESS',   '01,02,06'),                      
    ('25695', 'HARI',  'HARI ADDRESS',  '02,06'),                         
    ('89685', 'JEFF',  'JEFF ADDRESS',  '03,05,06,07'),                   
    ('47896', 'DAISY', 'DAISY ADDRESS', '03');
    

    查询 1:

    select SM.ROLLNO,
           SM.NAME, 
           SM.ADDRESS,
           (
           select ','+CM.CourseName
           from dbo.CourseMaster as CM
           where ','+SM.Course+',' like '%,'+CM.CourseId+',%'
           for xml path(''), type
           ).value('substring(text()[1], 2)', 'varchar(max)') as Course
    from dbo.StudentMaster as SM;
    

    结果:

    | ROLLNO |  NAME |       ADDRESS |          Course |
    |--------|-------|---------------|-----------------|
    |  12345 |   RAM |   RAM ADDRESS |     ABC,DEF,PQR |
    |  25695 |  HARI |  HARI ADDRESS |         DEF,PQR |
    |  89685 |  JEFF |  JEFF ADDRESS | GHI,MNO,PQR,STU |
    |  47896 | DAISY | DAISY ADDRESS |             GHI |
    
    • 5
  3. keb
    2017-02-25T12:39:13+08:002017-02-25T12:39:13+08:00

    从表中获取列表值的一种简单方法StudentMaster(您可以根据结果加入)是(借助前面提到的拆分字符串回复,并假设该函数为每个列表项返回一个名为 item 的列) :

    SELECT ROLLNO, NAME, ADDRESS, item
    FROM StudentMaster
    CROSS APPLY fSplitString(Course);
    
    • 0
  4. user158308
    2018-08-13T06:17:44+08:002018-08-13T06:17:44+08:00
    select D.DATE,D.DESCP,
       (
       select ','+S.NAME
       from tb_schems as S
       where ',' + D.SCHME + ',' like '%,' + cast(S.id as nvarchar(20)) + ',%'
       for xml path(''), type
       ).value('substring(text()[1], 2)', 'varchar(max)') as SCHEME
    from TB_DETAIL as D;
    
    • 0

相关问题

  • INNER JOIN 和 OUTER JOIN 有什么区别?

  • 什么时候应该使用唯一约束而不是唯一索引?

  • 死锁的主要原因是什么,可以预防吗?

  • JOIN 语句的输出是什么样的?

  • 如何确定是否需要或需要索引

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    连接到 PostgreSQL 服务器:致命:主机没有 pg_hba.conf 条目

    • 12 个回答
  • Marko Smith

    如何让sqlplus的输出出现在一行中?

    • 3 个回答
  • Marko Smith

    选择具有最大日期或最晚日期的日期

    • 3 个回答
  • Marko Smith

    如何列出 PostgreSQL 中的所有模式?

    • 4 个回答
  • Marko Smith

    列出指定表的所有列

    • 5 个回答
  • Marko Smith

    如何在不修改我自己的 tnsnames.ora 的情况下使用 sqlplus 连接到位于另一台主机上的 Oracle 数据库

    • 4 个回答
  • Marko Smith

    你如何mysqldump特定的表?

    • 4 个回答
  • Marko Smith

    使用 psql 列出数据库权限

    • 10 个回答
  • Marko Smith

    如何从 PostgreSQL 中的选择查询中将值插入表中?

    • 4 个回答
  • Marko Smith

    如何使用 psql 列出所有数据库和表?

    • 7 个回答
  • Martin Hope
    Jin 连接到 PostgreSQL 服务器:致命:主机没有 pg_hba.conf 条目 2014-12-02 02:54:58 +0800 CST
  • Martin Hope
    Stéphane 如何列出 PostgreSQL 中的所有模式? 2013-04-16 11:19:16 +0800 CST
  • Martin Hope
    Mike Walsh 为什么事务日志不断增长或空间不足? 2012-12-05 18:11:22 +0800 CST
  • Martin Hope
    Stephane Rolland 列出指定表的所有列 2012-08-14 04:44:44 +0800 CST
  • Martin Hope
    haxney MySQL 能否合理地对数十亿行执行查询? 2012-07-03 11:36:13 +0800 CST
  • Martin Hope
    qazwsx 如何监控大型 .sql 文件的导入进度? 2012-05-03 08:54:41 +0800 CST
  • Martin Hope
    markdorison 你如何mysqldump特定的表? 2011-12-17 12:39:37 +0800 CST
  • Martin Hope
    Jonas 如何使用 psql 对 SQL 查询进行计时? 2011-06-04 02:22:54 +0800 CST
  • Martin Hope
    Jonas 如何从 PostgreSQL 中的选择查询中将值插入表中? 2011-05-28 00:33:05 +0800 CST
  • Martin Hope
    Jonas 如何使用 psql 列出所有数据库和表? 2011-02-18 00:45:49 +0800 CST

热门标签

sql-server mysql postgresql sql-server-2014 sql-server-2016 oracle sql-server-2008 database-design query-performance sql-server-2017

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve