0

我需要对行进行两次更新,但我需要确保它们一起完成,并且其他用户的其他查询不会干扰它们。我知道,SELECT...FOR UPDATE但我想在第一次更新后它当然会被解锁,这意味着有人可能会干扰第二次更新。如果其他人先更新该行,则更新将起作用,但会弄乱数据。有没有办法确保这两个更新按照他们应该的方式发生?我被告知有关事务,但据我所知,它们仅有助于确保两个更新实际发生,而不是它们是否“一起”发生,除非我弄错了并且行将被锁定直到事务提交?

以下是查询:

SELECT z FROM table WHERE id='$id'

UPDATE table SET x=x+2 WHERE x>z

UPDATE table SET y=y+2 WHERE y>z

我犯了一个错误,没有提供完整的信息。那是我的错。我已经更新了查询。我遇到的问题是 z 也可以更新。如果 z 在 SELECT 之后但在其他两个更新之前更新,则数据可能会混乱。进行事务 BEGIN/COMMIT 是否适用?

4

3 回答 3

0

了解事务 http://dev.mysql.com/doc/refman/5.0/en/commit.html

[... connect ...]

mysql_query("BEGIN");

$query1 = mysql_query('UPDATE table SET x=x+2 WHERE x>y');
$query2 = mysql_query('UPDATE table SET y=y+2 WHERE y>y');

if($query1 && $query2) {
    mysql_query("COMMIT");  
    echo 'Save Done. All UPDATES done.';  
} else {
    mysql_query("ROLLBACK");  
    echo 'Error Save. All UPDATES reverted, and not done.';  
}
于 2012-04-08T03:54:59.440 回答
0

有各种级别的事务,但基本上根据ACID属性,您应该期望在给定事务中,所有读取和更新都是一致执行的,这意味着它将保持在有效状态,但更重要的是,事务在该工作中是隔离的在另一个事务(线程)中完成不会干扰您的事务(您的选择和更新 SQL 语句的分组):这允许您广泛假设您是系统内唯一允许您提交该组的执行线程工作(原子地)或全部回滚。

每个数据库可能会以不同的方式处理语义(有些可能会锁定行或列,有些可能会重新排序,有些可能会序列化),但这就是声明式数据库接口的美妙之处:您担心要完成的工作。

如前所述,在 MySQL 上,InnoDB 是事务性的,将支持上面提到的内容,因此请确保您的表是使用 InnoDB 组织的,其他非事务性引擎(例如 MyISAM)不是事务性的,因此将迫使您手动管理这些事务语义(锁定) .

于 2012-04-08T04:08:46.603 回答
0

一种方法是锁定整个表:

LOCK TABLE `table` WRITE;
SELECT z FROM `table` WHERE id='$id';
UPDATE `table` SET x=x+2 WHERE x>z;
UPDATE `table` SET y=y+2 WHERE y>z;
UNLOCK TABLES;

table这将防止其他会话在 SELECT 和 UPDATE 期间从表中写入和读取。

这是否是一个合适的解决方案确实取决于会话等待从表中读取或写入的合适程度。

于 2012-04-08T07:44:04.133 回答