3

我正在尝试从我的查询中分析独特的模式,并显示该模式是否出现在另一列中。

例如。

从我的查询中我得到:

----------
| Date | Value|
----------
|  1   |  0.8 |
----------
|  2   |  0.2 |
----------
|  3   |  0.4 |
----------
|  4   |  0.6 |
----------
|  5   |  0.3 |

我想检查是否有 3 个点在上升/下降并得到如下结果:

----------
| Date | Value| Ascending |
----------
|  1   |  0.8 |     0     |
----------
|  2   |  0.2 |     1     |
----------
|  3   |  0.4 |     1     |
----------
|  4   |  0.6 |     1     |
----------
|  5   |  0.3 |     0     |

升序列只是一点点,形成该模式的所有值都变为真。

有任何想法吗?

4

1 回答 1

8

以下代码不是完整的答案。它只比较两个数字,一个与下一个。

您需要与下一个值进行比较。假设“日期”列实际上是数字并且没有孔的升序(正如您在问题中所建议的那样),那么以下应该在两个数据库中都有效:

select t.*,
       (case when tnext.value > t.value then 1 else 0 end) as "ascending"
from t left outer join
     t tnext
     on t."date" = tnext."date" - 1;

如果要比较 3 行,则需要更多解释。我能想到的没有任何定义适用于您的结果。

编辑:

嗯,你似乎想找到升序,并标记至少有三行的那些。如果是这样,这有点难。下面使用以下技术枚举序列。它识别序列的开始,然后计算每个值的开始数。

要识别升序序列的开头,请查找值下降的对:

select t.*,
       (select 1
        from t tprev
        where tprev."date" = t."date" - 1 and
              tprev.value >= t.value
      ) as IsAscStart
from t;

为每个序列赋予其自己的 id:

select t.*
       (select count(*)
        from t t1 join
             t tprev
             on tprev."date" = t1."date" - 1 and
                tprev.value > t1.value
        where t1."date" <= t."date"
       ) as AscGrouping
from t;

从这一点开始的挑战是获得序列的长度。SQL Server 和 MySQL 之间的解决方法差异很大,所以我想不出一种简单的、独立于数据库的方法来做到这一点。

编辑二:

在 SQL Server 中,您可以执行以下操作:

with cte as (
    select t.*
           (select count(*)
            from t t1 join
                 t tprev
                 on tprev."date" = t1."date" - 1 and
                    tprev.value > t1.value
            where t1."date" <= t."date"
           ) as AscGrouping
    from t
   )
select t."date", t.value,
       (case when seqlen >= 3 then 1 else 0 end) as "Ascending"
from (select cte.*, count(*) over (partition by AscGrouping) as seqlen
      from cte
     ) t
于 2013-09-06T19:43:07.870 回答