在SQL中使行条目水平对齐

时间:2013-05-03 14:51:14

标签: sql sql-server-2008

所以这个问题类似于我之前提出的问题,但略有不同。

我正在查看被接纳并退出计划的客户的数据。对于每次入院和出院,他们都会进行评估并对​​其进行评分,有时他们会被录取并在一段时间内多次出院。

我需要能够将每个客户的承认分数与他们的下列出院日期配对,这样我就可以看到所有提高了一定金额的客户,从承认他们的每个承认和解雇的费用。

这是我现在如何格式化数据结果的虚拟示例: Before

这就是我理想的格式化方式: enter image description here

但我会在正确的方向或类似的格式帮助中采取任何一点,这样我就可以比较所有客户的所有入院和出院分数。

谢谢!

3 个答案:

答案 0 :(得分:2)

为了获得结果,您可以同时应用UNPIVOT和PIVOT函数。 UNPIVOT会将您的多列datescore转换为行,然后您可以将这些行转回列。

然后,unpivot语法将类似于:

select person,
  casenumber,
  ScoreType+'_'+col col,
  value,
  rn
from
(
  select person,
    casenumber,
    convert(varchar(10), date, 101) date,
    cast(score as varchar(10)) score,
    scoreType,
    row_number() over(partition by casenumber, scoretype
                      order by case scoretype when 'Admit' then 1 end, date) rn            
  from yourtable
) d
unpivot
(
  value
  for col in (date, score)
) unpiv

SQL Fiddle with Demo。这给出了一个结果:

| PERSON | CASENUMBER |             COL |      VALUE | RN |
-----------------------------------------------------------
|    Jon |       3412 |  Discharge_date | 01/03/2013 |  1 |
|    Jon |       3412 | Discharge_score |         12 |  1 |
|     Al |       3452 |      Admit_date | 05/16/2013 |  1 |
|     Al |       3452 |     Admit_score |         15 |  1 |
|     Al |       3452 |  Discharge_date | 08/01/2013 |  1 |
|     Al |       3452 | Discharge_score |         13 |  1 |

正如您所看到的,此查询还会创建新列以进行透视。所以最终的代码是:

select person, casenumber,
  Admit_Date, Admit_Score, Discharge_Date, Discharge_Score
from
(
  select person,
    casenumber,
    ScoreType+'_'+col col,
    value,
    rn
  from
  (
    select person,
      casenumber,
      convert(varchar(10), date, 101) date,
      cast(score as varchar(10)) score,
      scoreType,
      row_number() over(partition by casenumber, scoretype
                        order by case scoretype when 'Admit' then 1 end, date) rn

    from yourtable
  ) d
  unpivot
  (
    value
    for col in (date, score)
  ) unpiv
) src
pivot
(
  max(value)
  for col in (Admit_Date, Admit_Score, Discharge_Date, Discharge_Score)
) piv;

SQL Fiddle with Demo。这给出了一个结果:

| PERSON | CASENUMBER | ADMIT_DATE | ADMIT_SCORE | DISCHARGE_DATE | DISCHARGE_SCORE |
-------------------------------------------------------------------------------------
|     Al |       3452 | 05/16/2013 |          15 |     08/01/2013 |              13 |
|  Cindy |       6578 | 01/02/2013 |          17 |     03/04/2013 |              14 |
|  Cindy |       6578 | 03/04/2013 |          14 |     03/18/2013 |              12 |
|    Jon |       3412 |     (null) |      (null) |     01/03/2013 |              12 |
|  Kevin |       9868 | 01/18/2013 |          19 |     03/02/2013 |              15 |
|  Kevin |       9868 | 03/02/2013 |          15 |         (null) |          (null) |
|   Pete |       4765 | 02/06/2013 |          15 |         (null) |          (null) |
|  Susan |       5421 | 04/06/2013 |          19 |     05/07/2013 |              15 |

答案 1 :(得分:1)

SELECT 
ad.person, ad.CaseNumber, ad.Date as AdmitScoreDate, ad.Score as AdmitScore,
dis.date as DischargeScoreDate, dis.Score as DischargeScore
From
yourTable ad, yourTable dis

WHERE ad.person=dis.person and ad.ScoreType='Admit' and d is.ScoreType='Discharge';

答案 2 :(得分:1)

如果您提到的所有列都在同一个表中,您可以加入同一个表

 SELECT t1.person,
  t1.caseNumber,
   t1.date adate,
  t1.score  ascore,
  t1.scoreType ascoreType,
  t2.date ddate,
  t2.score  dscore,
  t2.scoreType dscoretype
FROM patient t1
join patient t2
on t1.casenumber=t2.casenumber
and t1.scoreType!=t2.scoreType
and t1.scoreType='Admit'

但这并不会显示已被录取但尚未出院的人的记录。我不知道你是否也在寻找这些信息。

SQL小提琴link

希望这有帮助!