我需要在查询中找到记录的页码。我有偏移和限制,但也需要行号来计算。
给定查询对象和记录ID,我怎样才能找到行号?
行号应该与查询表相关。之后将适用限制和抵消。
任何帮助表示赞赏...
答案 0 :(得分:2)
让记录的“页码”像这样倒退有点尴尬,通常一个系统想要从记录的“细节”反弹到“所有这些”的分页视图“随身携带页码到“详细”页面。
但是假设您正在使用一个功能强大的数据库,您可以使用window function从任意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()方法/函数提供窗口函数,因此SQLA查询假定典型的ORM映射如下:
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()
至于使用窗口函数与限制/偏移进行分页,值得查看我在这里写的comparison比较各种方法的性能,以及windowed range query配方I有时使用。
答案 1 :(得分:0)
我看起来很长很难,6小时后找到了解决方案。
我使用Mysql所以我相信我能做到这一点的唯一方法是使用@ variables。
以下是我将其实现为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()
来返回列表
答案 2 :(得分: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.
如果您还需要页码,那么只需更改上面的第1行:
SELECT d.myRowSerial, FLOOR((d.myRowSerial-1)/10) AS pageNumber
-- Say, 10 is per page;
第1页的pageNumber == 0,第2页的pageNumber = 1,.....