Разбить ряд чисел на равные части SQL Sever - PullRequest
1 голос
/ 16 апреля 2019

У меня следующий запрос, чтобы разделить диапазон чисел на части. Например, у меня есть диапазон 200 номеров, и я должен сгруппировать номера в 95, но результат приходит в обратном порядке. Я приложил ожидаемый результат

declare @MinValue bigint = 1 
declare @MaxValue bigint = 200;

declare @RowsPerGroup bigint =95
declare @RowsPerGroup1 bigint =(@RowsPerGroup-1)
;with src(val,rm) as (
select @MaxValue, (@MaxValue - @RowsPerGroup1) union all
select rm-1, case when rm-1 > @MinValue + @RowsPerGroup1 then rm-1 - @RowsPerGroup1 else @MinValue end from src where rm-1 >= @MinValue
)
select rm as 'Start', val  as 'End',[Difference]=(val-rm)+1 from src order by rm asc
option(maxrecursion 0)

Текущий результат:

Start   End Difference
1       10  10
11     105  95
106    200  95

Ожидаемый результат:

enter image description here

Пожалуйста, дайте мне знать, где я делаю не так

Ответы [ 2 ]

3 голосов
/ 16 апреля 2019

Мой вариант:

DECLARE
  @MinValue bigint = 1,
  @MaxValue bigint = 200,
  @RowsPerGroup bigint = 95 -- 300

;WITH cte AS(
  SELECT @MinValue [Start],IIF(@RowsPerGroup>@MaxValue,@MaxValue,@RowsPerGroup) [End]

  UNION ALL

  SELECT [End]+1,IIF([End]+@RowsPerGroup>@MaxValue,@MaxValue,[End]+@RowsPerGroup)
  FROM cte
  WHERE [End]<@MaxValue
)
SELECT [Start],[End],[End]-[Start]+1 [Difference]
FROM cte
1 голос
/ 16 апреля 2019

Вы можете увидеть причину в первой строке общего табличного выражения:

select @MaxValue, (@MaxValue - @RowsPerGroup1)

Это вставит (200,106) intto src.Второй выбор затем отсчитывает от существующих строк.Чтобы адаптировать CTE, обменивайте минимумы с максимумами (в том числе для диапазонов), инвертируйте арифметику, обратные сравнения и любые другие связанные обмены:

    select @MinValue, (@MinValue + @RowsPerGroup1) union all
    select val+1, case
        when val+1 < @MaxValue - @RowsPerGroup1
          then val+1 + @RowsPerGroup1
        else @MaxValue
        end
      from src
      where val+1 <= @MaxValue

Этот конкретный оператор можно упростить по частям:

    select @MinValue, (@MinValue + @RowsPerGroup - 1) union all
    select val + 1, case
        when val + @RowsPerGroup < @MaxValue
          then val + @RowsPerGroup
        else @MaxValue
        end
      from src
      where val < @MaxValue
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...