Warning: file_get_contents(/data/phpspider/zhask/data//catemap/8/mysql/71.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Mysql 查询中记录的行号_Mysql_Sqlalchemy - Fatal编程技术网

Mysql 查询中记录的行号

Mysql 查询中记录的行号,mysql,sqlalchemy,Mysql,Sqlalchemy,我需要在查询中查找记录的页码。我有偏移量和限制,但也需要行号来计算 给定一个查询对象和一个记录ID,我如何才能找到行号 行号应与查询表相对。之后将应用限制和偏移 感谢您的任何帮助…像这样将记录的“页码”向后移动有点尴尬,通常一个想要从记录的“细节”返回到“所有记录的分页视图”的系统只会将页码带到“细节”页面 但是,假设您使用的是一个功能强大的数据库,那么您可以使用一个数据库从任意SELECT语句中获取行号,这可以通过诸如Postgresql、SQL Server或Oracle(特别是MySQL或

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

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

行号应与查询表相对。之后将应用限制和偏移


感谢您的任何帮助…

像这样将记录的“页码”向后移动有点尴尬,通常一个想要从记录的“细节”返回到“所有记录的分页视图”的系统只会将页码带到“细节”页面

但是,假设您使用的是一个功能强大的数据库,那么您可以使用一个数据库从任意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\u 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通过方法/函数提供窗口函数,因此假设此方法的典型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()

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

我仔细研究了很久,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()
返回一个列表

经过长时间的搜索,我找到了以下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页的页码=0,第2页的页码=1,…

什么是“页码”?(我知道它是什么,我提供它作为一种方式让您思考您如何认为这是一个可回答的问题)相对于分页客户端的页码。与我正在寻找的答案不太相关,但我想我会提供一些背景。行数是否与表大小或页面大小等相关。。。等等您需要更具体地使用示例(我假设这里您有一个SQL查询)。您不只是将未分页查询作为一个子选择执行,并填写记录编号,然后根据单个记录的数据(特别是记录编号)和页面大小计算页码吗。。。嗯,没关系。我刚才看到了你的SQL炼金术标签。我的坏朋友。:-)行号是相对于查询表的。偏移量/限制将在以后应用。这将非常有效,但我使用的是MySQL,我应该在我的主题中指定它。我道歉。我找到了一个解决方案,我将发布。我很高兴你没有提到它是MySQL,否则我就无法提供一个关于窗口函数的优秀教程:)@zzzeek MySQL或SQLITE数据库怎么样?有什么解决方案吗?如果@rownum在本地建立到一个MySQL连接,那么您需要使用上面的显式连接(即使用engine.connect()作为conn:),因为从engine()执行不能保证使用相同的连接两次(加上并发访问可以获取该连接)。感谢您的解释。我觉得SQLALchemy文档有点难学(它非常段落化),你是怎么学的?啊,好吧。。。检查我的个人资料。我刚和一个想要更多段落的家伙在邮件列表上争论完。没有什么能取悦所有人。。。。