简体   繁体   English

排序desc后,mysql中的下一个和前一个记录

[英]Next and previous records in mysql after sorting desc

I have a simple table called users with the following data: 我有一个名为users的简单表,其中包含以下数据:

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

I want to make a prev/next navigation according to hops sorted descending. 我想根据跳跃排序降序进行上一个/下一个导航。 I use the following to query to sort descending: 我使用以下查询来降序排序:

SELECT * FROM users ORDER BY hops DESC, id DESC

This is the result: 这是结果:

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

Now what I want is that when I input any id in a mysql query I get the previous and next ids according to sorting above. 现在我想要的是,当我在mysql查询中输入任何id时,我会根据上面的排序得到上一个和下一个id。 For example: 例如:

For id 5 (In this case id=5 has the highest hops so no previous records before it): 对于id 5(在这种情况下,id = 5具有最高的跃点,因此之前没有先前的记录):

id (current) | hops (current) | id (prev) | hops (prev) | id (next) | hops (next)
   5         |    6           |  NULL     | NULL        |  6        |  5

For id 6: 对于id 6:

id (current) | hops (current) | id (prev) | hops (prev) | id (next) | hops (next)
   6         |    5           |  5        |     6       |  3        |  5

For id 3: 对于身份3:

id (current) | hops (current) | id (prev) | hops (prev) | id (next) | hops (next)
   3         |    5           |  6        |    5        |  1        |  3

For id 1: 对于id 1:

id (current) | hops (current) | id (prev) | hops (prev) | id (next) | hops (next)
   1         |    3           |  3        |    5        |  4        |  2

For id 4: 对于id 4:

id (current) | hops (current) | id (prev) | hops (prev) | id (next) | hops (next)
   4         |    2           |  1        | 3           |  2        |  1

For id 2 (In this case id=2 has the lowest hops so no next records after it) 对于id 2(在这种情况下,id = 2具有最低的跳数,因此之后没有下一个记录)

id (current) | hops (current) | id (prev) | hops (prev) | id (next) | hops (next)
   2         |    1           |  4        | 2           |  NULL     |  NULL

Thanks 谢谢

Try: 尝试:

select cp.*, n.id id_next, n.hops hops_next
from
(select c.id id_current, c.hops hops_current, p.id id_previous, p.hops hops_previous
 from
 (select * from users where id = ?) c
 left join users p on c.hops < p.hops or (c.id < p.id and c.hops = p.hops)
 order by p.hops, p.id limit 1) cp 
left join users n 
       on cp.hops_current > n.hops or (cp.id_current > n.id and cp.hops_current = n.hops)
order by n.hops desc, n.id desc limit 1

(SQLFiddle here ) (SQLFiddle 这里

This is the weirdest users table I've seen so far. 这是迄今为止我见过的最奇怪的用户表。 Anyway here's one way (although I have to admit it's a little complicated)... 无论如何这里是一种方式(虽然我不得不承认它有点复杂)......

DROP TABLE IF EXISTS test;

CREATE TABLE test(id INT NOT NULL AUTO_INCREMENT PRIMARY KEY,hops INT NOT NULL);

INSERT INTO test VALUES
(1 ,3),(2 ,1),(3,5),(4 ,2),(5 ,6),(6 ,5);

SELECT c.id id_curr
     , c.hops hops_curr
     , p.id id_prev
     , p.hops hops_prev
     , n.id id_next
     , n.hops hops_next
  FROM
     (
       SELECT a.*
            , COUNT(*) new_rank
         FROM 
            ( SELECT x.*
                   , COUNT(*) rank 
                FROM test x 
                JOIN test y 
                  ON y.hops >= x.hops 
               GROUP 
                  BY x.id
            ) a
         JOIN 
            ( SELECT x.*
                   , COUNT(*) rank 
                FROM test x 
                JOIN test y 
                  ON y.hops >= x.hops 
               GROUP 
                  BY x.id
            ) b
           ON b.rank < a.rank
           OR (b.rank = a.rank AND b.id >= a.id)
        GROUP 
           BY a.id
     )c
  LEFT
  JOIN
     (
       SELECT a.*
            , COUNT(*) new_rank
         FROM 
            ( SELECT x.*
                   , COUNT(*) rank 
                FROM test x 
                JOIN test y 
                  ON y.hops >= x.hops 
               GROUP 
                  BY x.id
            ) a
         JOIN 
            ( SELECT x.*
                   , COUNT(*) rank 
                FROM test x 
                JOIN test y 
                  ON y.hops >= x.hops 
               GROUP 
                  BY x.id
            ) b
           ON b.rank < a.rank
           OR (b.rank = a.rank AND b.id >= a.id)
        GROUP 
           BY a.id
     ) p
    ON p.new_rank = c.new_rank-1
  LEFT
  JOIN
     (
       SELECT a.*
            , COUNT(*) new_rank
         FROM 
            ( SELECT x.*
                   , COUNT(*) rank 
                FROM test x 
                JOIN test y 
                  ON y.hops >= x.hops 
               GROUP 
                  BY x.id
            ) a
         JOIN 
            ( SELECT x.*
                   , COUNT(*) rank 
                FROM test x 
                JOIN test y 
                  ON y.hops >= x.hops 
               GROUP 
                  BY x.id
            ) b
           ON b.rank < a.rank
           OR (b.rank = a.rank AND b.id >= a.id)
        GROUP 
           BY a.id
     ) n
    ON n.new_rank = c.new_rank+1
 ORDER 
    BY c.hops DESC
     , c.id DESC;

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM