5

一連の日付スパンのギャップを見つける方法は何ですか?

たとえば、次の日付スパンがあります。

1/ 1/11 - 1/10/11  
1/13/11 - 1/15/11  
1/20/11 - 1/30/11

次に、開始日と終了日を1/7/11と1/14/11にします。

1/10/11と1/13/11の間にギャップがあり、開始日と終了日が不可能であることを伝えたいと思います。または、最初に発生したギャップまでの日付スパンのみを返したい。

これがSQLサーバーで実行できるのであれば、それは良いことです。

私はそれが日付スパンに収まるかどうかを調べるために各日付を調べることを考えていました...そうでない場合は、その日にギャップがあります。

4

3 に答える 3

2
  • 最後から2番目のコードブロックにジャンプします:*I want to be able to tell that between 1/10/11 and 1/13/11 there is a gap so the start and end date is* 不可能。
  • 次の最後のコードブロックにジャンプします。*I want to return only the datespans up to the first gap encountered.*

まず、ここで議論する仮想テーブルがあります

create table spans (date1 datetime, date2 datetime);
insert into spans select '20110101', '20110110';
insert into spans select '20110113', '20110115';
insert into spans select '20110120', '20110130';

これは、カレンダーのすべての日付を個別に一覧表示するクエリです。

declare @startdate datetime, @enddate datetime
select @startdate = '20110107', @enddate = '20110114'
select distinct a.date1+v.number
from spans A
inner join master..spt_values v
  on v.type='P' and v.number between 0 and datediff(d, a.date1, a.date2)
-- we don't care about spans that don't intersect with our range
where A.date1 <= @enddate
  and @startdate <= A.date2

このクエリを使用して、予想される日数に対してカレンダーの日数をカウントすることにより、ギャップがあるかどうかをテストできます。

declare @startdate datetime, @enddate datetime
select @startdate = '20110107', @enddate = '20110114'

select case when count(distinct a.date1+v.number)
    = datediff(d,@startdate, @enddate) + 1
    then 'No gaps' else 'Gap' end
from spans A
inner join master..spt_values v
  on v.type='P' and v.number between 0 and datediff(d, a.date1, a.date2)
-- we don't care about spans that don't intersect with our range
where A.date1 <= @enddate
  and @startdate <= A.date2
-- count only those dates within our range
   and a.date1 + v.number between @startdate and @enddate

これを行う別の方法は、@ startから@endまでのカレンダーを前もって作成し、この日付のスパンがあるかどうかを確認することです。

declare @startdate datetime, @enddate datetime
select @startdate = '20110107', @enddate = '20110114'
-- startdate+v.number is a day on the calendar
select @startdate + v.number
from master..spt_values v
where v.type='P' and v.number between 0
  and datediff(d, @startdate, @enddate)

-- run the part above this line alone to see the calendar
-- the condition checks for dates that are not in any span (gap)
  and not exists (
    select *
    from spans
    where @startdate + v.number between date1 and date2)

クエリは、日付範囲@startのギャップであるすべての日付を返します-@ end ATOP 1を追加して、ギャップがあるかどうかを確認できます

ギャップの前にあるすべてのレコードを返すには、クエリをより大きなクエリの派生テーブルとして使用します

declare @startdate datetime, @enddate datetime
select @startdate = '20110107', @enddate = '20110114'
select *
from spans
where date1 <= @enddate and @startdate <= date2 -- overlaps
  and date2 < ( -- before the gap
    select top 1 @startdate + v.number
    from master..spt_values v
    where v.type='P' and v.number between 0
      and datediff(d, @startdate, @enddate)
      and not exists (
        select *
        from spans
        where @startdate + v.number between date1 and date2)
    order by 1 ASC
)
于 2011-01-22T06:09:18.737 に答える
1

MySQLを想定すると、次のようなものが機能します。

select @olddate := null;

select start_date, end_date, datediff(end_date, @olddate) as diff, @olddate:=enddate
from table
order by start_date asc, end_date asc
having diff > 1;

基本的に:前の行のend_dateを@olddate変数にキャッシュしてから、その「古い」値をcurrelenddateで差分します。持つ句は、2つの行の差が1日を超えるレコードのみを返します。

免責事項:これはテストしていませんが、基本的なクエリ構造は機能するはずです。

于 2011-01-22T01:22:28.043 に答える
1

1/10/11と1/13/11の間にギャップがあり、開始日と終了日が不可能であることを伝えたいと思います。

あなたはこの質問をしていると思います:あなたのテーブルのデータは開始日と終了日の間にギャップがありますか?

1列のテーブルdate_spanを作成し、それに日付スパンを挿入しました。

開始日と終了日の間の日数をカウントし、同じ範囲のdate_spanの行数を比較することで、ギャップを特定できます。

select 
  date '2011-01-14' - date '2011-01-07' + 1 as elapsed_days,  
  count(*) from date_span 
where cal_date between '2011-01-07' and '2011-01-14';

戻り値

elapsed_days count    
--           --
8            6

それらは等しくないため、2011-01-07と2011-01-14の間のテーブル「date_span」にギャップがあります。あなたが何をしようとしているのかよくわからないので、ここでやめます。

于 2011-01-22T02:15:35.340 に答える