1

我需要在查询中找到记录的页码。我有偏移和限制,但也需要行号来计算。

给定一个查询对象和一个记录 ID,我如何找到行号?

行号应该是相对于查询表的。限制和偏移量将在之后应用。

任何帮助表示赞赏...

4

3 回答 3

2

像这样让记录的“页码”倒退有点尴尬,通常一个想要从记录的“细节”反弹回“所有这些的分页视图”的系统只会随身携带页码到“详细信息”页面。

但是假设您正在使用功能强大的数据库,您可以使用窗口函数从任意 SELECT 语句中获取行号,您可以使用 Postgresql、SQL Server 或 Oracle(尤其不是 MySQL 或 SQLite)等数据库来获取行号。

使用 PG 我们可以从一些数据开始:

test=> create table data(id SERIAL primary key, value varchar(20));
NOTICE:  CREATE TABLE will create implicit sequence "data_id_seq" for serial column "data.id"
NOTICE:  CREATE TABLE / PRIMARY KEY will create implicit index "data_pkey" for table "data"
CREATE TABLE
test=> insert into data (value) values ('d1'), ('d2'), ('d3'), ('d4'), ('d5'), ('d6'), ('d7'), ('d8'), ('d9'), ('d10'), ('d11'), ('d12'), ('d13'), ('d14'), ('d15'), ('d16');
INSERT 0 16

row_number()然后我们可以选择这些数据并使用窗口函数获得整数行数:

test=> select value, row_number() over (order by id) as rownum from data;
 value | rownum 
-------+--------
 d1    |      1
 d2    |      2
 d3    |      3
 d4    |      4
 d5    |      5
 d6    |      6
 d7    |      7
 d8    |      8
 d9    |      9
 d10   |     10
 d11   |     11
 d12   |     12
 d13   |     13
 d14   |     14
 d15   |     15
 d16   |     16
(16 rows)

在子查询中应用窗口数据,我们可以根据这个计数选择结果的切片:

test=> select value from (select value, row_number() 
     > over (order by id) as rownum from data) as sub where rownum between 5 and 10;
 value 
-------
 d5
 d6
 d7
 d8
 d9
 d10
(6 rows)

因此,如果您有记录“d14”且页面大小为 5,您可以这样做:

test=> select (rownum - 1) / 5 from (select value, row_number() 
     > over (order by id) as rownum from data) as sub where value='d14';
 ?column? 
----------
        2
(1 row)

SQLAlchemy 通过over()方法/函数提供窗口函数,因此假设一个典型的 ORM 映射的 SQLA 查询如下所示:

subq = session.query(
            Data.value, 
            func.row_number().over(order_by=Data.id).label('rownum')
       ).subquery()
pagenum = session.query((subq.c.rownum - 1) / 5).\
       filter(subq.c.value == 'd14').scalar()

至于使用窗口函数与限制/偏移进行分页,值得查看我在这里写的比较各种方法的性能,以及我有时使用的窗口范围查询配方。

于 2013-02-05T01:31:27.473 回答
0

我看了很长时间,6小时后找到了解决方案。

我使用 Mysql,所以我相信我能做到这一点的唯一方法是使用 @ 变量。

这是我将它实现到 SQLAlchemy 中的方法:

qry = ...
sel = select([qry.c.id, "@rownum := @rownum + 1 as rownumber"])
conn = sel.engine.connect()
conn.execute("SET @rownum:=0")
res = conn.execute(sel)

@rownum将随着每一行递增。qry如果我们愿意, 我们可以从第二行添加更多列。res是带有列的最终表rownumber,我们可以在其中调用res.fetchall()以返回列表

于 2013-02-05T02:47:01.353 回答
0

经过长时间的搜索,我找到了以下 MySQL 解决方案:

SELECT d.myRowSerial
FROM (
    SELECT *, @rownum:=@rownum + 1 AS myRowSerial 
    FROM myTable, (SELECT @rownum:=0) AS nothingButSetInitialValue 
    WHERE 1=1 -- Optional: filter if required, otherwise, omit this line;
    ORDER BY AnyColumn -- Apply the order you like; 
) d
WHERE d.myColumn = 'Anything'; -- If you like to limit it to only
-- for any specific row(s), similar to the *MAIN query.

如果您也需要页码,那么只需像这样更改上面的第一行:

SELECT d.myRowSerial, FLOOR((d.myRowSerial-1)/10) AS pageNumber
-- Say, 10 is per page;

第 1 页的 pageNumber==0,第 2 页的 pageNumber=1,......

于 2014-05-22T06:15:14.197 回答