如何在不使用 join 或 cte 的情况下在同一查询中使用动态生成的列

Posted

技术标签:

【中文标题】如何在不使用 join 或 cte 的情况下在同一查询中使用动态生成的列【英文标题】:How to use the dynamically generated column in the same query without using join or cte 【发布时间】:2020-09-21 19:00:52 【问题描述】:

我正在学习 SQL。我有一种情况,我必须使用我在运行时创建的同一列。该表是 患者,只有 2 列,分别是 dateOfCheckupduration 以分钟为单位。

+---------------+-----------------+
| dateOfCheckup |   duration      | 
+---------------+-----------------+
|   2020-05-28  |  30 min         | 
|   2020-05-29  |  30 min         |
|   2020-05-30  |  1 hour         | 
|   2020-06-03  |  1 hour 30 min  | 
|   2020-06-05  |  30 min         |
|   2020-07-21  |  1 hour         |
|   2020-07-22  |  1 hour 30 min  | 
|   2020-07-28  |  1 hour 30 min  | 
+---------------+-----------------+

现在我将在运行时再创建 1 列,即 minutes(只是将 duration 转换为整数值的总分钟数):

select dateOfCheckup, duration,
((case when duration like '% hour%' then substring_index(duration, ' hour', 1) * 60 else 0 end) + (case when duration like '%min%' then substring_index(substring_index(duration, ' min', 1), ' ', -1) + 0 else 0 end)) as minutes from patient;

此查询运行良好。查看结果:

+---------------+-----------------+-----------------+
| dateOfCheckup |  duration       |   minutes       |
+---------------+-----------------+-----------------+
|   2020-05-28  |  30 min         |   30            | 
|   2020-05-29  |  30 min         |   30            |
|   2020-05-30  |  1 hour         |   60            |
|   2020-06-03  |  1 hour 30 min  |   90            |
|   2020-06-05  |  30 min         |   30            |
|   2020-07-21  |  1 hour         |   60            |
|   2020-07-22  |  1 hour 30 min  |   90            |
|   2020-07-28  |  1 hour 30 min  |   90            |
+---------------+-----------------+-----------------+

我的问题是,如果我想在同一个查询中使用这个新创建的列 minutes 来处理其他任务,例如 group by、order by、sum、avg 等。我只是询问是否有这样的场景,那么我们如何实现这一点。我尝试使用sum(minutes)。错误是:

“字段列表”中的未知列“分钟”

这是db<>fiddle。请帮帮我。

【问题讨论】:

你希望输出是什么? 例如,在另一列中设为分钟+10。 提供了下面的解决方案,如果对你有用的话 【参考方案1】:

您必须将当前查询视为子查询,然后对结果求和

SELECT SUM(minutes) AS total_time 
FROM (
    select dateOfCheckup, duration, 
    ((case when duration like '% hour%' then substring_index(duration, ' hour', 1) * 60 else 0 end) + 
    (case when duration like '%min%' then substring_index(substring_index(duration, ' min', 1), ' ', -1) + 0 else 0 end)
    ) as minutes 
from patient
) AS a;

【讨论】:

【参考方案2】:

您不能在同一查询的另一个表达式中使用列别名。

在子查询中定义别名,以便您可以在外部查询中引用它们:

SELECT t.dateOfCheckup, SUM(t.minutes)
FROM (
    select dateOfCheckup, duration,
      (
       (case when duration like '% hour%' 
             then substring_index(duration, ' hour', 1) * 60 
             else 0 end) 
     + (case when duration like '%min%' 
             then substring_index(substring_index(duration, ' min', 1), ' ', -1) + 0 
             else 0 end)
      ) as minutes 
    from patient
) AS t
GROUP BY t.dateOfCheckup

【讨论】:

这就是我想听到的“你不能在同一个查询的另一个表达式中使用列别名。” 你能看看这个吗:***.com/questions/64025684/…【参考方案3】:

我不知道你想做什么确切的操作,但使用一级子查询你可以以任何你想要的方式使用这个新的派生列,例如

select dateOfCheckup,duration,max(minutes) over() maxminutes
  from (select dateOfCheckup, duration, ((case when duration like '% hour%' then substring_index(duration, ' hour', 1) * 60 else 0 end) + (case when duration like '%min%' then substring_index(substring_index(duration, ' min', 1), ' ', -1) + 0 else 0 end)) as minutes 
          from patient
       ) p;

【讨论】:

以上是关于如何在不使用 join 或 cte 的情况下在同一查询中使用动态生成的列的主要内容,如果未能解决你的问题,请参考以下文章

在不使用 JOIN 的情况下在 HSQLDB 中进行透视

如何在不重新加载的情况下在同一页面上提交文件[重复]

如何在不更改构建系统的情况下在 C 项目中使用 rust 代码?

如何在不使用数据集的情况下在 Matlab 上比较不同的面部图像?

在不使用 boost::promise 的情况下在 boost 线程中返回值

如何在不使用原生 ios 应用程序中的 FBSDKShareDialog 的情况下在 Facebook 墙上分享照片或链接