3

我有一个名为 users 的简单表,其中包含以下数据:

id | hops
 1 | 3
 2 | 1
 3 | 5
 4 | 2
 5 | 4
 6 | 5

我想选择我指定的任何给定 id 的跃点数,并根据从最高到最低排序的跃点数选择下一个和上一个 id。

为了解释更多,我使用以下查询:

SELECT * FROM test WHERE id = 1 
OR (id > 1 AND hops >= (SELECT hops FROM test WHERE id= 1) ) 
OR (id < 1 AND hops <= (SELECT hops FROM test WHERE id= 1) ) 
LIMIT 3

因此,在上面的查询中,我尝试获取id=1具有相同或更高跳数的下一个 id,以及具有相同或更少跳数的前一个 id。

这是我得到的结果:

id | hops
1  | 3
3  | 5
5  | 4

正如你所看到的,它选择了 id=1 和两个更高的 id,尽管我只想要一个更高的 id 和一个更低的 id。因此,在这种情况下,结果应该是这样的:

id | hops
1  | 3
3  | 5

由于没有低于 1 的 id,所以没有低于 1 的 id 符合标准,只选择 1 个更高的 id。错误的结果是因为使用了 LIMIT 3 但我不能对每个条件都使用 LIMIT。所以根本不知道如何处理这个问题。

还有一个问题,使用子查询会 "SELECT hops FROM test WHERE id= 1" 大规模降低服务器速度吗?我听说使用子查询并不可取,但除了使用单独的查询外,没有其他方法可以得到这个数字。

谢谢

4

2 回答 2

3

给你,根据你的喜好改变ID的顺序......你没有说你想要最接近的跳数还是最接近的ID,只是更大或更小一个

SELECT * FROM test 
WHERE id IN (1,(
    SELECT id FROM test WHERE id > 1 AND hops >= (
        SELECT hops FROM test WHERE id = 1
        ) ORDER BY id LIMIT 1
    ), (
    SELECT id FROM test WHERE id < 1 AND hops <= (
        SELECT hops FROM test WHERE id = 1
        ) ORDER BY id DESC LIMIT 1
))
于 2013-04-01T19:59:53.853 回答
1

如果我正确理解您的问题,我相信以下内容会起作用。

-- Previous Rec
SELECT  t2.*
FROM    test as t1
        JOIN test as t2 ON t2.hop <= t1.id
WHERE   t1.id = 1
ORDER BY t2.id DESC
LIMIT 1

UNION ALL

-- Current Rec
SELECT  *
FROM    test as t
WHERE   id = 1

UNION ALL

-- Following Rec
SELECT  t2.*
FROM    test as t1
        JOIN test as t2 ON t2.id >= t1.hop
WHERE   t1.id = 1
ORDER BY t2.id ASC
LIMIT 1
于 2013-04-01T20:19:58.390 回答