3

ギャップがある可能性のある値を持つ ID という名前の整数列を持つテーブルがあります (例: 1,2,3,4,7,8,10,14,15,16,20)

上記の例で、次の結果になるクエリを見つけたいと思います。

1-4 7-8 10 14-16 20

=更新=

以下のコード (SQL Server でうまく動作するようです) のおかげで、目標である MS-Access でこれを動作させることに非常に近づいていると感じています。私は理解できない私のステートメントでまだ構文エラーが発生しています...

SELECT VAL FROM (

SELECT islands.PORTID, CSTR(islands.PORTID ) as val 
FROM MYTABLE islands
WHERE NOT EXISTS (SELECT * FROM MYTABLE t2 WHERE t2.PORTID = islands.PORTID - 1)
AND NOT EXISTS (SELECT * FROM MYTABLE t2 WHERE t2.PORTID = islands.PORTID + 1)

UNION 


SELECT 
    rngStart.PORTID
    ,CSTR(rngStart.PORTID ) + '-' 
        + CSTR(MIN(rngEnd.PORTID)) as val
FROM MYTABLE rngStart 
INNER JOIN MYTABLE checkNext ON checkNext.PORTID = rngStart.PORTID + 1
INNER JOIN
(
    SELECT PORTID 
    FROM MYTABLE tblRangeEnd 
    WHERE NOT EXISTS (SELECT * FROM MYTABLE t2 WHERE t2.PORTID = tblRangeEnd.PORTID + 1)
) rngEnd on rngEnd.PORTID > rngStart.PORTID 
WHERE NOT EXISTS (SELECT * FROM MYTABLE t2 WHERE t2.PORTID = rngStart.PORTID - 1)
GROUP BY rngStart.PORTID

) tbl として PORTID ASC で注文

4

4 に答える 4

0
create table #sequence (id int not null primary KEY)

insert into #sequence(id)
    select 1
    union all select 2
    union all select 3
    union all select 4
    union all select 7
    union all select 8
    union all select 10
    union all select 14
    union all select 15
    union all select 16
    union all select 20


--Find Contig ranges
select l.id as start,
    (
        select min(a.id) as id
        from #sequence as a
            left outer join #sequence as b on a.id = b.id - 1
        where b.id is null
            and a.id >= l.id
    ) AS fend
from #sequence as l
    left outer join #sequence as r on r.id = l.id - 1
where r.id is null;

--Find missing values in sequence
select l.id + 1 as start, min(fr.id) - 1 as stop
from #sequence as l
    left outer join #sequence as r on l.id = r.id - 1
    left outer join #sequence as fr on l.id < fr.id
where r.id is null and fr.id is not null
group by l.id, r.id;

drop table #sequence

これにより、範囲と範囲間のギャップの両方が得られるため、自分が持っているものや必要なものを確認できます。提供されたサンプルデータ

于 2011-11-28T19:06:13.183 に答える
0
SELECT val FROM
(
    -- Get the islands
    SELECT islands.ID, CAST(islands.ID as varchar(10)) as val 
    FROM @t1 islands
    WHERE NOT EXISTS (SELECT * FROM @t1 t2 WHERE t2.ID = islands.ID - 1)
    AND NOT EXISTS (SELECT * FROM @t1 t2 WHERE t2.ID = islands.ID + 1)

    UNION 
    
    -- Get the ranges
    SELECT 
        rngStart.ID
        ,CAST(rngStart.ID as varchar(10)) + '-' 
            + CAST(MIN(rngEnd.ID) as varchar(10)) as val
    FROM @t1 rngStart 
    INNER JOIN @t1 checkNext ON checkNext.ID = rngStart.ID + 1
    INNER JOIN
    (
        SELECT ID 
        FROM @t1 tblRangeEnd 
        WHERE NOT EXISTS (SELECT * FROM @t1 t2 WHERE t2.ID = tblRangeEnd.ID + 1)
    ) rngEnd on rngEnd.ID > rngStart.ID 
    WHERE NOT EXISTS (SELECT * FROM @t1 t2 WHERE t2.ID = rngStart.ID - 1)
    GROUP BY rngStart.ID
) as tbl
ORDER BY ID ASC

@t1 というテーブル変数を使用しましたが、それをテーブル名に置き換えてください。

ここでそれが実行されます。

編集

Access でこれを機能させるには、結合を少し変更する必要があります。これを試して:

SELECT val FROM
(
    SELECT islands.PORTID, CSTR(islands.PORTID) as val 
    FROM MYTABLE islands
    WHERE NOT EXISTS 
        (SELECT * FROM MYTABLE t2 WHERE t2.PORTID = islands.PORTID - 1)
    AND NOT EXISTS 
        (SELECT * FROM MYTABLE t2 WHERE t2.PORTID = islands.PORTID + 1)

    UNION 

    SELECT 
         rngStart.PORTID
        ,CSTR(rngStart.PORTID) + '-' + CSTR(MIN(endPORTID)) as val
    FROM MYTABLE rngStart 
    INNER JOIN 
    (
        SELECT checkNext.PORTID as nextPORTID, rngEnd.PORTID as endPORTID
        FROM MYTABLE checkNext
        INNER JOIN
        (
            SELECT rngEnd.PORTID
            FROM MYTABLE rngEnd 
            WHERE NOT EXISTS 
                (SELECT * FROM MYTABLE t2 WHERE t2.PORTID = rngEnd.PORTID + 1)
        ) AS rngEnd on rngEnd.PORTID > checkNext.PORTID - 1
    ) AS checkNext ON checkNext.nextPORTID = rngStart.PORTID + 1
    WHERE NOT EXISTS 
        (SELECT * FROM MYTABLE t2 WHERE t2.PORTID = rngStart.PORTID - 1)
    GROUP BY rngStart.PORTID
) as tbl
ORDER BY PORTID ASC
于 2011-11-28T18:58:19.097 に答える
0

これは、 function にアクセスする場合に機能しますrow_number()

with C as
(
  select ID,
         ID - row_number() over(order by ID) as grp
  from YourTable
)
select min(ID) as MinID,
       max(ID) as MaxID
from C
group by grp

または、共通テーブル式の代わりにサブクエリを使用します。

select min(ID) as MinID,
       max(ID) as MaxID
from (select ID,
             ID - row_number() over(order by ID) as grp
      from YourTable) as C
group by grp

結果:

MinID       MaxID
----------- -----------
1           4
7           8
10          10
14          16
20          20

SQL Server で試してみてください https://data.stackexchange.com/stackoverflow/q/119411/

于 2011-11-28T19:09:05.277 に答える
0

Another try:

SELECT
    MIN(i), MAX(i)
FROM (
    select 
        i - (SELECT COUNT(*) FROM tbl t WHERE t.i < tbl.i) g,
        i 
    from tbl
) t
GROUP BY g

It'll be slow for sure, but I don't see other way to number rows in access.

于 2011-11-28T18:53:13.160 に答える