sql - 在 Microsoft SQL Server 2000 中模拟 MySQL LIMIT 子句

标签 sql mysql sql-server zend-framework

当我在 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/

相关文章:

sql-server - 更改 SQL Server 2008 中现有表的列顺序/添加新列

sql - T-SQL 输出插入子句 - 访问不在插入/删除表中的数据

Mysql select into 语句以更容易理解的格式

mysql - 根据组从表中获取前 N 行

sql - 更新表时在 "FROM"或附近出现语法错误

mysql - 无法根据销售额对商品进行排名,

c# - 在 C# 中使用 IPv6 连接到 mySql 服务器

java - 将 Ignite 与 mysql 一起用作第 3 方持久存储时,如何确保缓存键和对象 id 相同

sql - 如何复制自引用表的行

sql - 查询转换表中的SQL