当我在 Zend Framework's database component 工作时,我们试图抽象出 MySQL、PostgreSQL 和 SQLite 支持的 LIMIT
子句的功能。也就是说,可以这样创建查询:
$select = $db->select();
$select->from('mytable');
$select->order('somecolumn');
$select->limit(10, 20);
当数据库支持 LIMIT
时,会产生如下 SQL 查询:
SELECT * FROM mytable ORDER BY somecolumn LIMIT 10, 20
这对于不支持 LIMIT
的数据库品牌来说更为复杂(顺便说一下,该子句不是标准 SQL 语言的一部分)。如果可以生成行号,请将整个查询设为派生表,并在外部查询中使用 BETWEEN
。这是 Oracle 和 IBM DB2 的解决方案。 Microsoft SQL Server 2005 也有类似的行号函数,所以可以这样写查询:
SELECT z2.*
FROM (
SELECT ROW_NUMBER OVER(ORDER BY id) AS zend_db_rownum, z1.*
FROM ( ...original SQL query... ) z1
) z2
WHERE z2.zend_db_rownum BETWEEN @offset+1 AND @offset+@count;
但是,Microsoft SQL Server 2000 没有 ROW_NUMBER()
函数。
所以我的问题是,你能想出一种方法来模拟 Microsoft SQL Server 2000 中的 LIMIT
功能,只使用 SQL 吗?不使用游标或 T-SQL 或存储过程。它必须支持 LIMIT
的两个参数,包括计数和偏移量。使用临时表的解决方案也是 Not Acceptable 。
编辑:
MS SQL Server 2000 最常见的解决方案似乎如下所示,例如获取第 50 到 75 行:
SELECT TOP 25 *
FROM (
SELECT TOP 75 *
FROM table
ORDER BY BY field ASC
) a
ORDER BY field DESC;
但是,如果总结果集为 60 行,则此方法不起作用。内部查询返回 60 行,因为它在前 75 行中。然后外部查询返回第 35-60 行,它不适合所需的 50-75“页面”。基本上,此解决方案有效,除非您需要结果集的最后一个“页面”,而该“页面”恰好不是页面大小的倍数。
编辑:
另一种解决方案效果更好,但前提是您可以假设结果集包含唯一的列:
SELECT TOP n *
FROM tablename
WHERE key NOT IN (
SELECT TOP x key
FROM tablename
ORDER BY key
);
结论:
似乎不存在用于在 MS SQL Server 2000 中模拟 LIMIT
的通用解决方案。如果您可以在 MS SQL 中使用 ROW_NUMBER()
函数,则存在一个很好的解决方案服务器 2005。
最佳答案
这是另一个仅适用于 Sql Server 2005 和更新版本的解决方案,因为它使用了 except 语句。但无论如何我都会分享它。 如果你想得到记录 50 - 75 写:
select * from (
SELECT top 75 COL1, COL2
FROM MYTABLE order by COL3
) as foo
except
select * from (
SELECT top 50 COL1, COL2
FROM MYTABLE order by COL3
) as bar
关于sql - 在 Microsoft SQL Server 2000 中模拟 MySQL LIMIT 子句,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/216673/