`

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

这个不行吗?

相关推荐

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

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

    MySQL命令大全

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

    mysql数据库my.cnf配置文件

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

    MYSQL常用命令大全

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

    MySQL 入门学习 ——基础教程

    当命令输入错误而又无法改变(多行语句情形)时,只要在分号出现前就可以用 c来取消该条命令 mysql&gt; select -&gt; user() -&gt; c mysql&gt; 这是一些最常用的最基本的操作命令,通过多次练习就可以牢牢掌捂了。 ...

    php mssql 数据库分页SQL语句

    根据自己使用过的内容,把常见数据库Sql Server,Oracle和MySQL的分页语句,从数据库表中的第M条数据开始取N条记录的语句总结如下: SQL Server 从数据库表中的第M条记录开始取N条记录,利用Top关键字:注意如果...

    基于springboot+mysql+ajax的网上借阅系统前端及后台源码+项目说明(高分毕设).zip

    在分类页面或搜索关键字后的图书展示页面,用户可以对图书进行排序操作,其中包括:按照综合排序、按照借阅量降序排序、按照价格升序 排序、 按照价格降序排序。 g.用户注册登录功能(在首页进行注册、登录操作) ...

    2009达内SQL学习笔记

    多数DBMS不需要在单条SQL语句后加分号,但特定的DBMS可能必须在单条SQL语句后加分号。 SQL语句的最后一句要以 “;”号结束 二、写子句顺序 Select column,group_function From table [Where condition] ...

    oracle学习文档 笔记 全面 深刻 详细 通俗易懂 doc word格式 清晰 连接字符串

    数值类型 Number(m,n) m(1~38) n(-84~127) 可以存储正数、负数、零、定点数和精度为38位的浮点数,其中,M表示精度,代表数字的总位数;N表示小数点右边数字的位数 日期类型 date 7字节 用于存储表中的日期和时间...

    UCenter Home 2.0 简体中文UTF8.zip

    您的MySQL数据库账号应当拥有CREATE、DROP、ALTER等执行权限,同时文件空间需不低于2M,数据库空间不低于5M,通常您的虚拟空间都会满足这个条件,以满足包括 UCenter Home在内的绝大多数网络软件的正常运行。...

    UCenter Home 2.0 简体中文GBK.zip

    您的MySQL数据库账号应当拥有CREATE、DROP、ALTER等执行权限,同时文件空间需不低于2M,数据库空间不低于5M,通常您的虚拟空间都会满足这个条件,以满足包括 UCenter Home在内的绝大多数网络软件的正常运行。...

    入门学习Linux常用必会60个命令实例详解doc/txt

    -n:一般而言,mount挂上后会在/etc/mtab中写入一笔资料,在系统中没有可写入文件系统的情况下,可以用这个选项取消这个动作。 4.应用技巧 在Linux 和Unix系统上,所有文件都是作为一个大型树(以/为根)的一部分...

    yershop开源网店系统 v3.8.2.zip

    yershop提供了积分、代金券、单品促销(直接打折、减价优惠、固定金额出售、买就赠代金券、买M件送N件)、订单促销(满额打折、满额优惠金额、满额送倍数积分、满额送优惠卷、满额免运费)、捆绑促销、团购、限时...

Global site tag (gtag.js) - Google Analytics