`

MySQL怎么查询排序后结果集的前[N, M]条记录

    博客分类:
  • SQL
阅读更多
今天群里有人提了这么一个问题,原话是:
“表T(a,b,c,d),怎么显示按c排序之后的21-30条记录。(a,b,c,d)中可能没有类似ID的属性”

我当时给出的答案是:
select * from T order by c where rownum in (21, 30);
不过后来想想可能这在MySQK上是不可行的。貌似只有Oracle支持吧。

先总结些流行数据库查询前N条记录的方法吧:
1、MySQL
;这里取N为5
select * from T order by c limit 5;
2、SQL Server:
select top 5 * from T order by c;
3、Oracle
select * from T order by c where rownum<5;

然后我针对MySQL给出了一条语句:
select A.* from T as A order by A.a limit 5 where A.a not in
(select B.a from T as B order by B.a limit 3);

这条语句在我现在这个版本的MySQL上执行不了,为什么呢?错误信息是:
ERROR 1235 (42000): This version of MySQL doesn't yet support 'LIMIT & IN/ALL/AN
Y/SOME subquery'


在MySQL4.1中子查询是不能使用LIMIT的,手册中也明确指明 This version of MySQL doesn’t yet support ‘LIMIT & IN/ALL/ANY/SOME subquery’

也就是说,这样的语句是不能正确执行的。
select * from table where id in (select id from table limit 10);

但是,只要你再来一层就行。。如:
select * from table where id in (select t.id from (select * from table limit 10)as t)

This version of MySQL doesn't yet support 'LIMIT & IN/ALL/ANY/SOME subquery'的意思是,这版本的 MySQL 不支持使用 LIMIT 子句的 IN/ALL/ANY/SOME 子查詢,即是支持非 IN/ALL/ANY/SOME 子查詢的 LIMIT 子查詢。
歧义点在于各生字的 apply 和組合顺序,这句子的正确結合顺序是 “This version of MySQL doesn’t yet support ((LIMIT-(IN/ALL/ANY/SOME)) subquery)” (”-”符号为hyphen ),有可能的错误理解是 “This version of MySQL doesn’t yet support ((LIMIT subquery) & ((IN/ALL/ANY/SOME) subquery))”。

所以最后我给出的答案是:
select * from T as C where C.c in (select p.* from (select A.c from T as A order by A.c limit 30) as p)
and C.c not in
(select q.* from (select B.c from T as B order by B.c limit 21) as q);


虽然答案有点长,但是执行效率还是不算低的。如果大家有什么更好的方法,欢迎拍砖和指正。

更正:
shiwell给出了很简单的答案,我当时对Limit的用法不是很会,这里学习了,顺便感谢shinwell。
他给的答案是:
select * from T order by c LIMIT 21, 30
2
0
分享到:
评论
5 楼 shansun123 2009-06-03  
zzxwill 写道


喜哥吧?你太有才了~
4 楼 zzxwill 2009-06-03  
楼下的楼下好像最简单!
3 楼 zzxwill 2009-06-03  
看了很受益~
下面是一些方法:

显示查询结果集的[M,N]结果
1. T(c),显示按c排序之后的21-30条记录
a) 方法一:子查询
mysql> select * from test
    -> ;
+-------+
| bonus |
+-------+
|     2 |
|  3000 |
|     0 |
|    80 |
|    50 |
|   150 |
+-------+
下面是加上“属性”rownum
mysql> select bonus, (select count(*) from test where t.bonus>=bonus ) as rownum
  from(select * from test order by bonus) t;
+-------+--------+
| bonus | rownum |
+-------+--------+
|     0 |      1 |
|     2 |      2 |
|    50 |      3 |
|    80 |      4 |
|   150 |      5 |
|  3000 |      6 |
+-------+--------+
6 rows in set (0.00 sec)

选出2到6条记录
mysql> select bonus from(select t.bonus, (select count(*) from test where t.bonu
s>=bonus ) as rownum  from(select * from test order by bonus) t) t2 where t2.row
num>=2 and t2.rownum<=5;
+-------+
| bonus |
+-------+
|     2 |
|    50 |
|    80 |
|   150 |
+-------+
4 rows in set (0.00 sec)

b) 方法二:ifnull函数
设置@x 初值:如
mysql> set @x=0;
Query OK, 0 rows affected (0.00 sec)
查询:
mysql>  select @x:=ifnull(@x,0)+1 as rownum , bonus from test order by bonus;
+--------+-------+
| rownum | bonus |
+--------+-------+
|      7 |     0 |
|      8 |     2 |
|      9 |    50 |
|     10 |    80 |
|     11 |   150 |
|     12 |  3000 |
+--------+-------+
6 rows in set (0.00 sec)

方法三:使用联接查询(笛卡尔积)
mysql> select a.bonus, count(*) as rownum  from test a,test b where a.bonus>=b.b
onus group by a.bonus;
+-------+--------+
| bonus | rownum |
+-------+--------+
|     0 |      1 |
|     2 |      2 |
|    50 |      3 |
|    80 |      4 |
|   150 |      5 |
|  3000 |      6 |
+-------+--------+
6 rows in set (0.00 sec)

2. T(c),显示21-30条记录
方法一:ifnull函数
mysql>  select @x:=ifnull(@x,0)+1 as rownum , bonus from test;
+--------+-------+
| rownum | bonus |
+--------+-------+
|      1 |     2 |
|      2 |  3000 |
|      3 |     0 |
|      4 |    80 |
|      5 |    50 |
|      6 |   150 |
+--------+-------+
6 rows in set (0.00 sec)

方法二:上面方法三
2 楼 shansun123 2009-06-03  
shinwell 写道

select * from T order by c LIMIT 21, 30

这个不行吗?

非常感谢您的留言,我试了,可以的。呵呵。
1 楼 shinwell 2009-06-03  
select * from T order by c LIMIT 21, 30

这个不行吗?

相关推荐

    SQL查询前10条记录(SqlServermysqloracle)语法分析.docx

    总结来说,SQL Server、MySQL和Oracle都有各自的方法来查询数据库中的前N条记录。理解并熟练掌握这些语法对于进行有效的数据库操作至关重要,尤其是在处理大量数据时的分页查询。通过调整相应的参数,我们可以轻松地...

    MySql基本查询、连接查询、子查询、正则表达查询讲解

    6、合并查询结果 7、为表和字段取别名 8、使用正则表达式查询 什么是查询? 怎么查的? 数据的准备如下: [sql] view plain copy create table STUDENT( STU_ID int primary KEY, STU_NAME char(10) not null, ...

    mysql 查询第几行到第几行记录的语句

    在MySQL中,`LIMIT` 子句用于限制查询返回的结果集数量,这在处理大量数据时非常有用,可以避免一次性加载过多数据导致性能下降。以下是对标题和描述中提到的知识点的详细说明: 1. **查询第一行记录**: 使用 `...

    mySql与oracle分页技术

    `LIMIT m, n`语句用于限制查询结果的数量,其中`m`是起始索引,`n`是返回的记录数。由于在MySQL中索引是从0开始的,所以如果我们要获取第二页的数据(每页显示10条记录),`LIMIT`的参数应该是`LIMIT 10, 10`。这将...

    MySQL大数据量分页查询方法及其优化

    此外,由于LIMIT是基于结果集的偏移量定位记录,因此每次查询的结果顺序可能会有所不同,这对于需要稳定排序的场景来说是不合适的。 #### 方法2:建立主键或唯一索引,利用索引 - **语句样式**:`SELECT * FROM 表...

    MySQL查询指令汇总

    - **查询第 3 页的数据,每页显示 5 条记录**: ```sql SELECT * FROM staffs LIMIT 5 OFFSET 10; ``` - **公式计算偏移量**: ```sql offset = (page - 1) * num; ``` ##### 6. 子查询 - **基本语法**: ...

    MySQL数据库:项目查询与维护学生信息管理数据表.pptx

    排序是指将查询结果按照一定的规则排序,例如按照学号排序: ```sql SELECT * FROM student ORDER BY sno; ``` 总结 本节学习目标是掌握简单数据查询的使用,包括单表无条件数据查询、单表有条件数据查询、聚集...

    mysql,mssql,oracle分页精典代码集锦

    - **解释**:此方法首先选择前 (M-1)*N 行的最大 ID,然后从表中选取 ID 大于这个值的前 N 条记录,并按 ID 排序。 - **示例**:如果要从第 6 行开始返回 5 条记录,可以写为: ```sql SELECT TOP 5 * FROM book...

    mysql数据库my.cnf配置文件

    #MySQL的查询缓冲大小(从4.0.1开始,MySQL提供了查询缓冲机制)使用查询缓冲,MySQL将SELECT语句和查询结果存放在缓冲区中, # 今后对于同样的SELECT语句(区分大小写),将直接从缓冲区中读取结果。根据MySQL用户...

    MySQL索引背后的数据结构及算法原理

    - **索引选择性**:选择性高的索引可以更快地过滤出结果集,因此在创建索引时应优先考虑选择性较高的字段。 - **前缀索引**:对于较长的字符串字段,创建前缀索引可以减少索引大小,但需要注意前缀长度的选择。 - **...

    MySQL命令大全

    例如,往表 MyClass中插入二条记录, 这二条记录表示:编号为的名为Tom的成绩为.45, 编号为 的名为Joan 的成绩为.99,编号为 的名为Wang 的成绩为.5. mysql&gt;insert into MyClass values(1,’Tom’,96.45),(2,’Joan...

    记录一些mysql的学习总结.zip

    - 视图是基于SQL查询的结果集,提供了一种逻辑上的表,可以隐藏数据的复杂性,简化用户接口。 5. 存储引擎: - InnoDB:支持事务处理,具有外键支持,是最常用的存储引擎。 - MyISAM:读写速度快,但不支持事务...

    如何在SQL Server中实现 Limit m,n 的功能

    `TOP n`用于选取查询结果的前n条记录。例如,如果你只想获取`tablename`表的前6条记录,你可以使用以下SQL语句: ```sql SELECT TOP 6 id FROM tablename ``` 然而,当需要获取特定范围的记录,如第7条到第9条时,...

    mysql数据库cmd命令大全

    7. 查询表记录:使用 `select *或字段列表 from 表名 where 条件 order by 排序字段;` 命令。 8. 删除表记录:使用 `delete from 表名 where 条件;` 命令。 9. 修改表记录:使用 `update 表名 set 字段=值 where ...

    sql语法(oracle,mysql,sqlserver)

    根据给定的信息,我们可以深入探讨SQL中的多表查询技术,特别是针对Oracle、MySQL以及SQL Server数据库系统的应用。这里我们将重点放在几个关键的概念上:笛卡尔积、连接(Join)的不同类型,尤其是内连接(Inner ...

Global site tag (gtag.js) - Google Analytics