【发布时间】:2010-09-18 00:24:00
【问题描述】:
当我处理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 的两个参数,包括计数和偏移量。使用临时表的解决方案也是不可接受的。
编辑:
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 Server 2005 中使用 ROW_NUMBER() 函数,则存在一个很好的解决方案。
【问题讨论】:
标签: sql mysql sql-server zend-framework