增加字段时如何确保MySQL数据库中没有竞争条件?

How to make sure there is no race condition in MySQL database when incrementing a field?(增加字段时如何确保MySQL数据库中没有竞争条件?)
本文介绍了增加字段时如何确保MySQL数据库中没有竞争条件?的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

当两个连接想要更新同一条记录时,如何防止 MySQL 数据库中的竞争条件?

How to prevent a race condition in MySQL database when two connections want to update the same record?

例如,连接 1 想要增加尝试"计数器.第二个连接也想做同样的事情.两个连接SELECT尝试"计数,增加值并且两个UPDATE尝试"增加值.突然间,尝试次数"只是尝试次数+1"而不是尝试次数+2",因为两个连接都得到了相同的尝试次数"并加了 1.

For example, connection 1 wants to increase "tries" counter. And the second connection wants to do the same. Both connections SELECT the "tries" count, increase the value and both UPDATE "tries" with the increased value. Suddenly "tries" is only "tries+1" instead of being "tries+2", because both connections got the same "tries" and incremented it by one.

如何解决这个问题?

推荐答案

这里有 3 种不同的方法:

Here's 3 different approaches:

update table set tries=tries+1 where condition=value;

它将以原子方式完成.

如果您确实需要先选择值并在您的应用程序中更新它,您可能需要使用事务.这意味着您必须使用 InnoDB,而不是 MyISAM 表.您的查询将类似于:

If you do need to first select the value and update it in your application, you likely need to use transactions. That means you'll have to use InnoDB, not MyISAM tables. Your query would be something like:

BEGIN; //or any method in the API you use that starts a transaction
select tries from table where condition=value for update;
.. do application logic to add to `tries`
update table set tries=newvalue where condition=value;
END;

如果交易失败,您可能需要手动重试.

if the transaction fails, you might need to manually retry it.

一种常见的方法是在表中引入一个版本列.您的查询将执行以下操作:

A common approach is to introduce a version column in your table. Your queries would do something like:

select tries,version from table where condition=value;
.. do application logic, and remember the old version value.
update table set tries=newvalue,version=version + 1 where condition=value and version=oldversion;

如果该更新失败/返回受影响的 0 行,则其他人同时更新了该表.您必须从头开始 - 也就是说,选择新值,执行应用程序逻辑并再次尝试更新.

If that update fails/returns 0 rows affected, someone else has updated the table in the mean time. You have to start all over - that is, select the new values, do the application logic and try the update again.

这篇关于增加字段时如何确保MySQL数据库中没有竞争条件?的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持编程学习网!

本站部分内容来源互联网,如果有图片或者内容侵犯您的权益请联系我们删除!

相关文档推荐

Hibernate reactive No Vert.x context active in aws rds(AWS RDS中的休眠反应性非Vert.x上下文处于活动状态)
Bulk insert with mysql2 and NodeJs throws 500(使用mysql2和NodeJS的大容量插入抛出500)
Flask + PyMySQL giving error no attribute #39;settimeout#39;(FlASK+PyMySQL给出错误,没有属性#39;setTimeout#39;)
auto_increment column for a group of rows?(一组行的AUTO_INCREMENT列?)
Sort by ID DESC(按ID代码排序)
SQL/MySQL: split a quantity value into multiple rows by date(SQL/MySQL:按日期将数量值拆分为多行)