【问题标题】:CONCATS for column names, column aliasesCONCATS 用于列名、列别名
【发布时间】:2017-01-18 09:41:07
【问题描述】:

我正在尝试为一个月内的每一天运行一个查询。本月第一天的查询部分如下所示,并且有效:

CASE 
WHEN CONCAT(
    Shifts.d1a,
    Shifts.d1b, 
    Shifts.d1c, 
    Shifts.d1d, 
    Shifts.d1e, 
    Shifts.d1f) = "" THEN "-"
ELSE
  CONCAT (
    IF (Shifts.d1a='Yes','A',''),
    IF (Shifts.d1b='Yes','B',''),
    IF (Shifts.d1c='Yes','C',''),
    IF (Shifts.d1d='Yes','D',''),
    IF (Shifts.d1e='Yes','E',''),
    IF (Shifts.d1f='Yes','F','')
    ) END AS d1,

该月的第二天看起来相同,只是表引用的所有“d1”部分都是“d2”,而“END AS d1”将是“END AS d2”。

我继承了表结构;每个 d[n][x] 的值为“Yes”或“”。不涉及空值。目标是在一天内获取单独字段的项目 - 即 d[n]a、d[n]b、d[n]c 等,并将它们连接到(例如)每个字段的“ABF”中值,如果所有字段为空,则在 d[n] 列中生成连字符。

就像我说的,如果我运行 31 次几乎相同的代码,这将有效。这让我觉得丑陋和不必要。但是我遇到了在 SELECT 中使用串联字符串作为列名以及在输出中使用变量作为列别名的一部分的明显障碍。所以这不起作用:

BEGIN
  DECLARE vcounter INT DEFAULT 0;
  DECLARE vcolumn VARCHAR (10);
  DECLARE vcolumna VARCHAR (64);
  DECLARE vcolumnb VARCHAR (64);
  DECLARE vcolumnc VARCHAR (64);
  DECLARE vcolumnd VARCHAR (64);
  DECLARE vcolumne VARCHAR (64);
  DECLARE vcolumnf VARCHAR (64);
shift_loop: REPEAT
SET vcounter = vcounter + 1;
SET vcolumn = CONCAT ('d',vcounter);
SET vcolumna = CONCAT ('Shift.',vcolumn,'a');
SET vcolumnb = CONCAT ('Shift.',vcolumn,'b');
SET vcolumnc = CONCAT ('Shift.',vcolumn,'c');
SET vcolumnd = CONCAT ('Shift.',vcolumn,'d');
SET vcolumne = CONCAT ('Shift.',vcolumn,'e');
SET vcolumnf = CONCAT ('Shift.',vcolumn,'f');
SELECT 
  CASE 
   WHEN CONCAT(
        vcolumna,
        vcolumnb,
        vcolumnc,
        vcolumnd,
        vcolumne,
        vcolumnf) = '' THEN '-'
    ELSE 
        CONCAT(
        vcolumna,
        vcolumnb,
        vcolumnc,
        vcolumnd,
        vcolumne,
        vcolumnf
        ) 
        END
          AS vcolumn;
UNTIL vcounter = 1
END REPEAT;
END

是的,我知道过程中的 CASE 语句是不必要的——我可以将最终 CONCAT 的结果分配给一个变量并针对一个空值对其进行测试。我只是保持原样,以便更好地了解我想要做什么。

结果是,对于每一行,我得到一个名为“vcounter”的列,其内容为“Shift.d1aShift.d1bShift.d1c”等。 有人有什么建议吗?

编辑添加

好的,谢谢,我查看了建议的答案,但我不知道如何使它与我需要的 WHILE 语句一起工作。这是我将代码更改为:

 BEGIN  
 SET @vcounter = 0;  
 SET @vcounter = @vcounter + 1;  
 SET @vtable = 'Table_X';  
 SET @vcolumn = CONCAT ('d',@vcounter);  
 SET @vcolumna = CONCAT (@vtable,'.',@vcolumn,'a');  
 SET @vcolumnb = CONCAT (@vtable,'.',@vcolumn,'b');  
 SET @vcolumnc = CONCAT (@vtable,'.',@vcolumn,'c');  
 SET @vcolumnd = CONCAT (@vtable,'.',@vcolumn,'d');  
 SET @vcolumne = CONCAT (@vtable,'.',@vcolumn,'e');   
 SET @vcolumnf = CONCAT (@vtable,'.',@vcolumn,'f');  
 SET @vshifts = CONCAT  
 (@vcolumna,',',@vcolumnb,',',@vcolumnc,',',@vcolumnd,',',@vcolumne,',',@vcolumnf);  
 SET @vquery = CONCAT ('SELECT CASE WHEN CONCAT (',  
                  @vshifts,  
                  ') ="" THEN "-" ELSE   
                  CONCAT (IF(',@vcolumna,' = "Yes","A",""),  
                          IF(',@vcolumnb,' = "Yes","B",""),  
                          IF(',@vcolumnc,' = "Yes","C",""),  
                          IF(',@vcolumnd,' = "Yes","D",""),  
                          IF(',@vcolumne,' = "Yes","E",""),  
                          IF(',@vcolumnf,' = "Yes","F","")  
                  )END AS ',@vcolumn,' FROM ',@vtable);  
 PREPARE stmt FROM @vquery;  
 EXECUTE stmt;  
 END  

这在一天内效果很好。问题是,在我插入 WHILE 或 REPEAT 语句的任何地方,我都会得到不好的结果。我找到的最接近的结果最终产生的结果看起来像

 d1  
 D  
 F  
 A  
 B  

 d2  
 A  
 AB  
 C  
 -  

等等。我想要的是

 d1    d2  
 D     A
 F     AB
 A     C
 B     -

我显然不太了解自己在这里做什么,所以我非常感谢任何人提供的任何帮助。谢谢。

【问题讨论】:

  • 虽然我没有看到您的真实表格结构,但您的表格设计似乎存在严重问题。你能换桌子吗?
  • 通过准备好的语句使用动态 sql 并考虑@juergend 的评论。
  • 抱歉,我继承了这个结构,为了向后兼容,我必须保留它。
  • 有时我想象世界上有一个开发人员实施了破坏 1NF 的糟糕数据库设计,然后他很快就离开去另一家公司工作,做同样的事情。

标签: mysql


【解决方案1】:

好的,我自己解决了。以下是有效的:

 BEGIN
 SET @vcounter = 0;
 SET @vtable = 'tablename';
 SET @vquery = 'SELECT
  last_name As Last_Name,
  first_name As First_Name, ';
 shift_loop: REPEAT
 SET @vcounter = @vcounter + 1;
 SET @vcolumn = CONCAT ('d',@vcounter);
 SET @vcolumna = CONCAT (@vcolumn,'a');
 SET @vcolumnd = CONCAT (@vcolumn,'b');
 SET @vcolumnp = CONCAT (@vcolumn,'c');
 SET @vcolumne = CONCAT (@vcolumn,'d');
 SET @vcolumnt = CONCAT (@vcolumn,'e');
 SET @vcolumns = CONCAT (@vcolumn,'f');
 SET @vshifts = CONCAT (@vcolumna,',',@vcolumnb,',',@vcolumnc,',',@vcolumnd,',',@vcolumne,',',@vcolumnf);
 SET @vquery = CONCAT (@vquery,' CASE WHEN CONCAT (',
                       @vshifts,
                       ') ="" THEN "-" ELSE 
                       CONCAT (IF(',@vcolumna,' = "Yes","A",""),
                              IF(',@vcolumnb,' = "Yes","B",""),
                              IF(',@vcolumnc,' = "Yes","C",""),
                              IF(',@vcolumnd,' = "Yes","D",""),
                              IF(',@vcolumne,' = "Yes","E",""),
                                  IF(',@vcolumnf,' = "Yes","F","")
                       ) END AS ',@vcolumn,',');
 Until @vcounter = 31
 END REPEAT;
 SET @vquery = CONCAT (
     @vquery,'
     comment AS Comment,
     _submitted_ AS Submitted
     FROM ',@vtable);
 PREPARE stmt FROM @vquery;
 EXECUTE stmt;
 DEALLOCATE PREPARE stmt;
 END

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2013-05-01
    • 1970-01-01
    • 2022-12-11
    • 2023-04-02
    • 1970-01-01
    • 1970-01-01
    • 2014-07-03
    相关资源
    最近更新 更多