SP插入后,下一页有一个空结果,直到重新加载
Posted
技术标签:
【中文标题】SP插入后,下一页有一个空结果,直到重新加载【英文标题】:After SP Insert, next page have an empty result until reloaded 【发布时间】:2012-08-21 07:02:44 【问题描述】:我有一个将数据插入数据库的存储过程(从现在开始为 SP)(SaveClient,见下文)。当 SP 完成后,我将 php 页面重定向到另一个列出条目的 PHP 页面(FetchObjectList,见下文)。在我重新加载/刷新页面之前,列表不会返回新创建的记录。
存储过程末尾有一个COMMIT
,调用SP后我在PHP代码中关闭了数据库连接,检查错误但没有出错。
页面本身返回一个 200 状态码,这意味着它没有被缓存,因此也不能与浏览器相关。
当前的解决方法是在 PHP 代码中使用 sleep(1)
,但是当代码上线时,我不知道它是否足够。我当然宁愿让 mysql 提供正确的结果集。
编辑:我正在使用 PHP 的 MySQLi 对象接口,了解一下可能会有用。 ;)
我的开发计算机安装了 PHP 5.2.17、MySQL 5.0.51a (InnoDB) 和 Apache 2.2.17,并在 Windows 7 x64 上运行。
更新
在 SaveClient 的末尾添加了以下行 CALL FetchObjectList('client_tbl', NULL, NULL, 1, 'client_tbl.name ASC', NULL, NULL);
。结果集中在呈现的结果集中没有新创建的客户端。
更新 2
我尝试使用SQL_NO_CACHE
,如here 所见,但无济于事。
我现在将直接在 PHP 中尝试相同的 SQL,而不是调用 SP。
9 月 3 日至 20 日更新
到目前为止,我已经尝试了任何合理的答案/评论,但没有任何运气。我今天尝试更新我的 PHP 和 MySQL 版本(因为我今天了解到实时服务器将在 PHP 5.3.something 和 MySQL 5.1.something 上运行)但没有让它工作。我需要更新 PHP 以获得更新的php_mysqli.dll
/libmysql.dll
,因为我得到的只支持高达 5.0.51a 并且可能存在我的问题,因为实际数据库中没有任何工作。我尝试了 MySQL 安装中的 libmysql.dll
无济于事。
请注意,我还更改了包含的 PHP 代码,因为我实际上复制了错误的调用 user_tbl
而不是 client_tbl
的代码,并且还对其进行了简化(删除了多查询),但结果仍然相同。
我不知道赏金会发生什么,如果它恢复给我,我会再次添加它。
存储过程SaveClient
DELIMITER //
DROP PROCEDURE IF EXISTS work.SaveClient//
CREATE PROCEDURE work.SaveClient(
IN ObjectID INT,
IN UserID INT,
IN ClientName VARCHAR(60),
IN VersionFrom DATETIME,
IN VersionTo DATETIME)
root:BEGIN
DECLARE EXIT HANDLER FOR SQLEXCEPTION ROLLBACK;
/*
Default values ---------------------------------------------------------------------------------------------------------
*/
# Used to block INSERT/UPDATEs
SET @DoChanges = TRUE;
SET @Fields = '*';
SET @Version = NULL;
SET @UserVersion = NULL;
SET @DateNow = NOW();
SET @VersionActive = CONCAT(
'( ( NOW() BETWEEN ',
'version_from AND ',
'version_to ) OR ( ',
'version_from < NOW() AND ',
'version_to IS NULL ) )'
);
IF VersionFrom IS NULL THEN
SET VersionFrom = @DateNow;
END IF;
/*
Search for client ------------------------------------------------------------------------------------------------------
*/
IF ObjectID IS NOT NULL THEN
SET @Client = CONCAT(
'SELECT version INTO @Version FROM client_tbl WHERE object_id = ',
ObjectID,
' AND ',
@VersionActive
);
PREPARE stmt FROM @Client;
EXECUTE stmt;
DEALLOCATE PREPARE stmt;
# Check if there are any changes
IF @Version IS NOT NULL THEN
SELECT name INTO @Name FROM client_tbl WHERE name = ClientName AND version = @Version;
IF @Name = ClientName THEN
SET @errorMsg = "Duplicate entry";
SET @errorCode = "S0000002";
SELECT @errorCode, @errorMsg;
LEAVE root;
END IF;
END IF;
END IF;
/*
Search for user ---------------------------------------------------------------------------------------------------------
*/
# Create this as a function
IF UserID IS NOT NULL THEN
SET @User = CONCAT(
'SELECT version INTO @UserVersion FROM user_tbl WHERE object_id = ',
UserID,
' AND ',
@VersionActive
);
PREPARE stmt FROM @User;
EXECUTE stmt;
DEALLOCATE PREPARE stmt;
END IF;
IF @UserVersion IS NULL THEN
SET @errorMsg = "User is missing";
SET @errorCode = "U0000099";
SELECT @errorCode, @errorMsg;
LEAVE root;
END IF;
/*
Add the client ---------------------------------------------------------------------------------------------------------
*/
# Close the current version
IF @Version IS NOT NULL THEN
IF @DoChanges = TRUE THEN
CALL UpdateVersion(
ObjectID,
UserID,
@Version,
@DateNow,
'client_tbl'
);
SET @Version = @Version + 1;
END IF;
ELSE
SET @Version = 1;
END IF;
IF @DoChanges = TRUE THEN
IF ObjectID IS NULL THEN
INSERT INTO
object_tbl
(
object_class_id,
created,
created_by
)
VALUES(
2,
NOW(),
UserID
)
;
SET ObjectID = LAST_INSERT_ID();
END IF;
INSERT INTO
client_tbl
(
object_id,
version,
version_from,
version_to,
changed,
changed_by,
name
)
VALUES(
ObjectID,
@Version,
VersionFrom,
NULL,
@DateNow,
UserID,
ClientName
)
;
END IF;
COMMIT;
END //
DELIMITER ;
存储过程 FetchObjectList
DELIMITER //
DROP PROCEDURE IF EXISTS work.FetchObjectList//
CREATE PROCEDURE work.FetchObjectList(
IN ObjectType VARCHAR(60),
IN ObjectSubType VARCHAR(60),
IN ObjectSubID INT,
IN IsActive INT,
IN OrderBy VARCHAR(100),
IN SetStart INT,
IN MaxResults INT)
root:BEGIN
DECLARE EXIT HANDLER FOR SQLEXCEPTION ROLLBACK;
# Allow the "JSON" output be a max of 8kb
SET GLOBAL group_concat_max_len = 8096;
/*
Default values ---------------------------------------------------------------------------------------------------------
*/
SET @Fields = '*';
SET @VersionWhere = '1'; # Get everything
SET @Special = '';
SET @OrderBy = '';
SET @SetStart = '';
SET @MaxResults = '';
SET @JoinIn = '';
IF IsActive = 1 THEN
SET @VersionWhere = CONCAT(
'( NOW() BETWEEN ',
ObjectType,
'.version_from AND ',
ObjectType,
'.version_to OR ( ',
ObjectType,
'.version_from < NOW() AND ',
ObjectType,
'.version_to IS NULL ) )'
);
END IF;
IF OrderBy != '' THEN
SET @OrderBy = CONCAT(
'ORDER BY ',
OrderBy
);
END IF;
/*
Specials for each type -------------------------------------------------------------------------------------------------
*/
/*
- Clients ------------
*/
IF ObjectType = 'client_tbl' THEN
SET @Fields = '
*,
client_tbl.object_id AS object_id,
(
SELECT
COUNT(*) AS Total
FROM
client_user_privilege_tbl cup
WHERE
cup.client_id = client_tbl.object_id
) AS usercount
';
END IF;
/*
- Configuration ------------
*/
IF ObjectType = 'configuration_tbl' THEN
SET @Fields = '
*
';
END IF;
/*
Add upp the query to run -----------------------------------------------------------------------------------------------
*/
SET @Query = CONCAT(
'SELECT ',
@Fields,
' FROM ',
ObjectType,
' ',
@JoinIn,
' WHERE ',
@VersionWhere,
' ',
@Special,
@OrderBy
);
PREPARE stmt FROM @Query;
EXECUTE stmt;
DEALLOCATE PREPARE stmt;
COMMIT;
END //
DELIMITER ;
PHP 代码片段(9 月 20 日更新)
$query = "CALL FetchObjectList('client_tbl', NULL, NULL, 1, NULL, NULL, NULL)";
addTrace($query);
$rs = $db->query($query);
if( $rs )
addTrace('Query done -> Results: ' . $rs->num_rows);
while($r = $rs->fetch_assoc())
$fetchArray[] = $r;
$count = $rs->num_rows;
$rs->close();
$db->next_result();
else
addTrace('Query failed -> ' . $db->error);
flushTrace();
exit;
【问题讨论】:
我有一个将数据保存到数据库的存储过程。当我在插入后加载的页面上运行另一个存储过程时(插入一次后加载新页面?或页面刚刚重新加载?),结果为空。如果我重新加载页面,结果就在那里。请编辑第一句话,因为您更清楚实际发生了什么 添加了我在 SaveClient SP 中使用 SP 的更新。结果相同,新记录不在结果集中,但仍存储在数据库中。 能否验证SP后数据库发生了变化? (使用 HeidiSQL 或 PhpMyAdmin 等)可能只是第二页的缓存问题。尝试添加随机 GET 参数。 是的。现在,您应该为清晰度投票。现在问题也可以理解了。只做一件事,你的问题就被抓住了。做什么?执行存储过程(您将自动转到重定向页面)不要刷新数据库中的页面检查值是否存在?他们会偏离轨道。 如果您在上述条件下找到值,那么您必须仅在通过存储过程获取值的地方共享重定向页面的 php 代码。实际上最有可能出现错误。 【参考方案1】:由于它是 mysql 的一个相当陈旧的版本,我不会感到惊讶,这与错误有关,但我会 - 在你的位置 - 想知道的一件事是这是否可以通过 not 使用交易。 (例如 autocommit = on)。
对于该版本 5.0,我还将检查查询缓存并将其全部禁用而不是每个查询(请参阅 SHOW VARIABLES LIKE 'have_query_cache'; SET GLOBAL query_cache_size =0; )。这至少会消除那些在这个问题中发挥作用的人,重现(或尝试)问题,看看是否有任何改变。如果没有,我会开始搜索特定的错误,特别是当查询缓存被禁用并且它仍然在不使用事务的情况下执行此操作时。
我验证了对 5.0 mysql (innodb) 的支持。
innodb_flush_log_at_trx_commit = 1 innodb_flush_method = O_DIRECT在您的 my.cnf 中专门设置这些选项,最重要的一项。他们很好地解释了那些做什么。
见http://dev.mysql.com/doc/refman/5.0/en/innodb-parameters.html#sysvar_innodb_flush_log_at_trx_commit
【讨论】:
我已经禁用了查询缓存并从 SP 中删除了所有事务 SQL,检查了autocommit
是否为 ON,并且我还有 innodb_flush_log_at_trx_commit=1
。 O_DIRECT
在 Windows 上不可用。
确实,我的错,这确实不是 Windows 选项。虽然涉及多个结果集,但 Amal Jacob 的观点是有道理的。
我没有来自SaveClient
和FetchObjectList
的多个结果集。 SaveClient
仅在我发现重复条目时返回结果集,FetchObjectList
仅返回一个。在网页本身上,我现在将其限制为仅运行一个查询/呼叫,但仍然存在问题。这不是你的坏事,你正在努力帮助我,我很感激。
此时我将打开通用日志并查看实际执行/返回的内容。 dev.mysql.com/doc/refman/5.0/en/query-log.html 这个问题让我很感兴趣。我认为 Amal Jacob 仍然是对的,我记得过去遇到过这个问题,就像他提到的那样,SP 本身有一个隐藏的结果集,所以你的 + 一个空的 = multi ,你应该可以在一般日志中看到阿法克。
我看到了:120923 17:42:07 30 在数据库上连接 user@localhost 30 查询 CALL FetchObjectList('configuration_tbl', NULL, NULL, NULL, NULL, NULL, NULL) 30 查询 CALL SaveClient (NULL,1,'TuffTuff', NULL, NULL) 让我感兴趣的是它不像所有其他页面那样说 30 Quit。我认为我们在这里有所作为。【参考方案2】:
mysql_query 的问题是它不支持多个结果集。这正是您的存储过程所做的——它们往往返回多个结果集。每当您调用 SP 时,退出状态都会秘密地携带在(空)结果集中。如果您将其与您自己的过程输出相加,则查询中的某些结果集将被您的 PHP 检索代码忽略。当您尝试运行另一个查询时,待处理的结果集仍将在缓冲区中。
【讨论】:
我没有使用mysql_query
。我正在使用mysqli
扩展名。
这不就是你应该做$mysqli->next_result();
的原因吗?如果是这样,它已经在我认为需要的每一个环节都完成了。以上是关于SP插入后,下一页有一个空结果,直到重新加载的主要内容,如果未能解决你的问题,请参考以下文章
php 表单 - 2 个按钮(一个用于销毁会话并重新加载页面,一个用于进入下一页)