PHP Conference Japan 2024

mysql_num_rows

(PHP 4, PHP 5)

mysql_num_rows获取结果中的行数

警告

此扩展在PHP 5.5.0中已弃用,并在PHP 7.0.0中移除。应改用MySQLiPDO_MySQL扩展。另请参见MySQL:选择API指南。此函数的替代方案包括

描述

mysql_num_rows(资源 $result): int|false

检索结果集中的行数。此命令仅对返回实际结果集的语句(如SELECT或SHOW)有效。要检索INSERT、UPDATE、REPLACE或DELETE查询影响的行数,请使用mysql_affected_rows()

参数

result

正在评估的结果资源。此结果来自对mysql_query()的调用。

返回值

成功时返回结果集的行数,失败时返回false

范例

示例 #1 mysql_num_rows() 示例

<?php

$link
= mysql_connect("localhost", "mysql_user", "mysql_password");
mysql_select_db("database", $link);

$result = mysql_query("SELECT * FROM table1", $link);
$num_rows = mysql_num_rows($result);

echo
"$num_rows 行\n";

?>

注释

注意:

如果使用mysql_unbuffered_query(),则在检索结果集中的所有行之前,mysql_num_rows()将不会返回正确的值。

注意:

为了向后兼容性,可以使用以下已弃用的别名:mysql_numrows()

参见

添加注释

用户贡献的注释 17 条注释

12
sam at liddicott dot com
20 年前
此页面上有一些用户评论,以及一些资源,包括位于

http://www.faqts.com/knowledge_base/view.phtml/aid/114/fid/12 的常见问题解答

这不是一个特别通用的解决方案,阅读此页面评论的人也应该意识到

如果在查询中使用“group by”或“having”,则 select count(*) 可能不会给出正确的结果,因为 count(*) 是一个聚合函数,并且每次 group-by 列更改时都会重置。

select sum(..) ... left join .. group by ... having ...

可以在 mysql 3 中作为子查询的替代方案,并且此类查询不能将 select 字段替换为 count(*) 以获得良好的结果,它就是不起作用。

Sam
5
pjoe444 at yahoo dot com
20 年前
回复我的上一条条目

这似乎是让“普通”循环运行的最佳解决方法,并且可以根据行号更改输出
(例如,安排计划)

$rowno=mysql_num_rows($result);

for ($i=0; $i<mysql_num_rows($result); $i++) {
$row = mysql_fetch_assoc($result);

print "<div class=\"showing\">";
print "<b>".$row['timeon']."-".$row['timeoff']."</b> ".$row['event']."<br />;
if ($i!=$rowno-1) {
print "other-html-within-sched-here</div>";
}
else print "end-last-entry-html-here</div>";
} //关闭循环
5
aaronp123 att yahoo dott comm
21 年前
我可能是唯一一个遇到这种情况的人——但是,如果你的 MyISAM 表有一行,并且你使用有效的表和列名搜索你可能期望为 0 行的结果,你将不会得到 0,你将得到 1,这是 MyISAM 在表有 0 行或 1 行时的优化响应。“5.2.4 MySQL 如何优化 WHERE 子句”中写道

*及早检测无效的常量表达式。MySQL 会快速检测到某些 SELECT 语句是不可能的,并返回没有行。



*所有常量表都在查询中的任何其他表之前读取。常量表是
1) 空表或只有一行数据的表。
2) 使用 WHERE 子句在 UNIQUE 索引或 PRIMARY KEY 上使用的表,其中所有索引部分都与常量表达式一起使用,并且索引部分定义为 NOT NULL。

希望这可以防止有人整夜都在处理 1146 错误,除非我完全误解了我已经弄清楚了这一点。
2
Joebomb
9 年前
回复上一篇文章:这可能并不总是能正确工作,因为 $object->doesExist 将包含结果,而不是布尔值。更好的方法(使用相同的方法)是使用强制转换

<?php
class Object {
var
$doesExist = false;

[...]
function
load() {
$result = mysql_query('...');
$this->doesExist = (bool) ($res = mysql_fetch_array($result))
[...]
}
}
?>
2
philip at cornado dot c()m
23 年前
关于 SQL count(),请参阅此常见问题解答
* http://www.faqts.com/knowledge_base/view.phtml/aid/114/fid/12
注意:如果你已经有 $result,请在其上使用 mysql_num_rows(),否则使用 SQL count()。不要仅仅为了计数而 SELECT 数据。
1
匿名用户
19 年前
回复 oran at trifeed dot com

你之所以会遇到这种行为,是因为你没有为你的 FOUND_ROWS() 结果指定别名

$qry = mysql_query ( 'SELECT FOUND_ROWS() AS total' );
$rst = mysql_fetch_array ( $qry, MYSQL_ASSOC );
echo $rst['total'];

Sean :)
-1
Typer85 at gmail dot com
17年前
关于以下用法的说明;建议使用多个MySQL函数来获取表记录的数量。

你可能熟悉以下方法

<?php

$sqlQuery
= 'Select SQL_CALC_FOUND_ROWS `MyField` From `MyTable` Limit 1;';

$sqlQuery_1 = 'Select FOUND_ROWS( );';

?>

我省略了实际连接到MySQL和执行查询的部分,但是你应该明白我的意思。

我做了一些测试,在一个相当高流量的网站上,一个经常执行多个查询的网站,发现使用这种MySQL函数组合实际上会导致错误的结果。

例如,假设我有两个查询来获取两个不同表中表记录的数量。因此,实际上,我们正在执行4个查询(每个表2个查询)。

如果两个不同的请求通过PHP进入,你将遇到问题。请注意,当我说请求时,我的意思是两个不同的客户端请求你的PHP页面。

---------------
请求1
---------------

执行:在表1上执行SQL_CALC_FOUND_ROWS

---------------
请求2
---------------

执行:在表2上执行SQL_CALC_FOUND_ROWS

---------------
请求1
---------------

执行:Select FOUND_ROWS( )

此时,你看到了发生的竞争条件。在请求1执行时,请求2进来了。

此时,请求1将返回表2中表记录的数量,而不是预期的表1!

为什么?因为MySQL不区分请求。每个查询都在队列中等待轮到它。一旦轮到它,它就会被MySQL执行。

MySQL函数Select FOUND_ROWS( )将返回最后一个SQL_CALC_FOUND_ROWS的结果!

记住这一点。
-2
dzver at abv dot bg
19 年前
运行第二个查询“select count(...) from ...”比在第一个查询中添加SQL_CALC_FOUND_ROWS,然后使用select FOUND_ROWS() + mysql_num_rows()更快。
-3
pjoe444 at yahoo dot com
20 年前
很遗憾,似乎没有办法在典型的循环中获取当前正在迭代的行号,
例如
while ($row = mysql_fetch_assoc($result)) { }

毕竟,这是一个行数组的数组,如
mysql_num_rows($result)

所示,假设这给出“40行”:知道迭代是在第39行时会很有用。

最接近的是“data seek”:但它直接连接到
行号,例如(来自mysql_data_seek页面)

for ($i = mysql_num_rows($result) - 1; $i >= 0; $i--) {
if (!mysql_data_seek($result, $i)) {
echo "Cannot seek to row $i: " . mysql_error() . "\n";
continue;
}

它仍然无法解决在普通循环中知道你正在处理哪一行的行号问题。

这种情况的一个原因是php fetch(一次fetch一行)的构造,没有任何合理的FOR循环可能性以及行号。

建议
$Rows[$i] 的可能性,其中
$i 将是行号

$Rows[$row[], $row[], $row[].....]
0 1 2 等

-- 在行内优秀的检索($row[$i]),
虽然肯定更重要,但与
行本身的类似可能性不匹配。

当然,Count($result)不起作用,$result只是一个
简单的票证标识符……

Peter T
-4
gtaylor at sonic dot net
16年前
在防止SQL_CALC_FOUND_ROWS和FOUND_ROWS()操作的竞争条件方面,将FOUND_ROWS()结果包含在实际结果集中可能会变得复杂且有些笨拙,特别是对于复杂的查询和/或结果排序。查询变得更复杂,你可能难以隔离/排除FOUND_ROWS()结果,而mysql_num_rows()将返回实际结果数+1,所有这些都使你的代码更混乱,更难阅读。但是,竞争条件是真实存在的,必须处理。

一种替代的更清晰的方法是使用WRITE锁显式锁定表(防止其他进程读取/写入表)。我能看到的缺点是性能下降,并且你的mysql用户必须具有锁定权限。

<?php
// 请原谅使用了mysqli而不是mysql

$mysqli->query("LOCK TABLE t WRITE");
$results = $mysqli->query("SELECT id FROM t LIMIT 0,10");
$totalNumResults = array_pop($mysqli->query("SELECT FOUND_ROWS()")->fetch_row());
$mysqli->query("UNLOCK TABLES");
?>

现在你可以像其他结果集一样迭代结果。
-4
oran at trifeed dot com
19 年前
对我来说

SELECT SQL_CALC_FOUND_ROWS 与
SELECT FOUND_ROWS()

一起使用,只有以下语法有效:
$result = @mysql_query($query);
$resultTotal = @mysql_query("SELECT FOUND_ROWS()");
$res= mysql_fetch_array($resultTotal);
echo $res['FOUND_ROWS()'];

希望对您有所帮助

oran
http://www.trifeed.com
-3
wil1488 at gmail dot com
19 年前
使用SQL COUNT函数,无需选择源……

参见示例

<?
//与数据库建立连接

$my_table = mysql_query("SELECT COUNT(*) as TOTALFOUND from table", $link); //执行SQL代码
注意:将在TOTALFOUND中返回总数

print (mysql_result($my_table,0,"TOTALFOUND")); //使用字段营从SQL查询中获取总数!
?>

谢谢,祝你好运。
-4
Typer85 at gmail dot com
14年前
实际上,我有点不好意思这么说,但我对2007年7月17日06:44发布的一条相当旧的笔记进行了更正。

使用SQL_CALC_FOUND_ROWS和FOUND_ROWS( )不会在MySQL上触发竞争条件,因为这几乎会违背它们的全部目的。

它们用法的结果实际上对于每个连接会话都是唯一的,因为进程不可能在PHP中共享任何内容。就PHP而言,每个请求都表示与MySQL的新连接,因为每个请求都隔离到其自己的进程中。

要模拟这一点,请创建以下脚本

<?php

$Handle
= mysql_connect( "localhost" , "root" , "" );
mysql_select_db( "lls" );

if( isset(
$_GET[ 'Sleep' ] ) ) {
mysql_query( "SELECT SQL_CALC_FOUND_ROWS `bid` From `blocks` Limit 1" );
} else {
mysql_query( "SELECT SQL_CALC_FOUND_ROWS `aid` From `access` Limit 1" );
}

if( isset(
$_GET[ 'Sleep' ] ) ) {
sleep( 10 ); // 模拟另一个HTTP请求进来。
$Result = mysql_query( "SELECT FOUND_ROWS( )" );
print_r( mysql_fetch_array( $Result ) );
}

mysql_close( );

?>

设置与你的环境匹配的连接和查询信息。

使用Sleep查询字符串运行一次脚本,再次不使用它运行一次。同时运行它们很重要。使用Apache ab或类似的东西,或者更容易,只需打开两个浏览器标签页。例如

https://127.0.0.1/Script.php?Sleep=10
https://127.0.0.1/Script.php

如果存在竞争条件,则脚本第一次实例的结果将等于脚本第二次实例的结果。

例如,脚本的第二个实例将执行以下SQL查询

<?php

mysql_query
( "SELECT SQL_CALC_FOUND_ROWS `aid` From `access` Limit 1" );

?>

当脚本的第一个实例休眠时,会发生这种情况。如果存在竞争条件,当脚本的第一个实例唤醒时,它执行的FOUND_ROWS( )的结果应该是脚本第二个实例执行的SQL查询的行数。

但是当你运行它们时,情况并非如此。脚本的第一个实例返回其自身查询的行数,即

<?php

mysql_query
( "SELECT SQL_CALC_FOUND_ROWS `bid` From `blocks` Limit 1" );

?>

因此,事实证明不存在竞争条件,并且为解决此“问题”而提出的所有解决方案实际上都是不需要的。

祝你好运,
-5
tac at smokescreen dot org
22年前
MySQL 4.0支持一项很棒的新功能,允许你获取如果查询没有LIMIT子句将返回的行数。要使用它,你需要在查询中添加SQL_CALC_FOUND_ROWS,例如

$sql = "Select SQL_CALC_FOUND_ROWS * from table where state='CA' limit 50";
$result = mysql_query($sql);

$sql = "Select FOUND_ROWS()";
$count_result = mysql_query($sql);

你现在有了表中与条件匹配的总行数。这对于在浏览列表时了解记录总数非常有用。
-5
webmaster dasourcerer net
17年前
在我的一个应用程序中,我必须让一个对象知道它是否存在于数据库中。我找到了一种无需使用mysql_num_rows()的廉价解决方案

<?php
class Object {
var
$doesExist = false;

[...]
function
load() {
$result = mysql_query('...');
$this->doesExist = ($res = mysql_fetch_array($result))
[...]
}
}
?>
-5
simon_nuttall at hotmail dot com
19 年前
wil1488 at gmail dot com关于计算表行的评论的面向对象版本
<?php
$result
= $mysqli->query("SELECT COUNT(*) as TOTALFOUND from table");
$row_array=$result->fetch_array(MYSQLI_ASSOC);
print(
$row_array['TOTALFOUND']);
?>
-5
Tom Fejfar
16年前
对 chrisdberry82 at gmail dot com 代码的改进

<?php
$sql
= "
SELECT SQL_CALC_FOUND_ROWS
'0', z.id
FROM
zoom AS z
LIMIT
0,6
UNION
SELECT
'1', FOUND_ROWS()
ORDER BY `0` DESC , RAND()"
;
?>

您可以看到,您可以按照任何您喜欢的顺序排列最终结果;)
然后您可以像这样获取结果
<?php
$res
= mysql_query($sql);
$count = mysql_fetch_assoc($res);
while(
$row = mysql_fetch_assoc($res)){
unset(
$row["0"]); //去除“排序列”
print_r($row); //或其他任何操作;)
}
echo
$count["id"]; //总行数
?>

并且您去除了糟糕的 IF 语句;)
To Top