Select xxx for update

一. Mysql 数据库 的 RR 隔离级别下, 如果在事务中使用SELECT ... FOR UPDATE, 实现如下伪代码所描述的效果:

begin transaction
select id from t where status = false limit 1,1;
if select result is not empty;
then update status = true where id = ;
commit;

那么能实现吗? 答案是肯定的。

其实在MySQL数据库的可重复读(Repeatable Read,RR)隔离级别下,通过事务使用SELECT ... FOR UPDATE可以实现伪代码所描述的效果。以下是详细说明和实现步骤:

伪代码描述

BEGIN;
SELECT id FROM t WHERE status = false LIMIT 1 FOR UPDATE;
IF (SELECT result is not empty) THEN
    UPDATE t SET status = true WHERE id = <the selected id>;
COMMIT;

实现步骤

在MySQL中,可重复读(RR)隔离级别可以防止幻读、脏读和不可重复读。在RR隔离级别下,使用SELECT ... FOR UPDATE可以确保选中的行在事务提交前不会被其他事务修改。

SQL 实现

  1. 开启事务

    START TRANSACTION;
    
  2. 选择并锁定符合条件的行

    SELECT id FROM t WHERE status = false LIMIT 1 FOR UPDATE;
    
  3. 检查查询结果并更新

    • 在编程语言中(如Java、Python等)处理查询结果,如果结果不为空,则进行更新操作。
    • 假设使用的是Java,示例代码如下:

Java示例

假设我们使用JDBC来实现上述逻辑,以下是一个示例代码:

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;

public class MySQLTransactionExample {
    public static void main(String[] args) {
        String url = "jdbc:mysql://localhost:3306/yourdatabase";
        String user = "yourusername";
        String password = "yourpassword";

        Connection conn = null;
        PreparedStatement selectStmt = null;
        PreparedStatement updateStmt = null;
        ResultSet rs = null;

        try {
            // 获取数据库连接
            conn = DriverManager.getConnection(url, user, password);
            conn.setAutoCommit(false); // 开启事务

            // 执行SELECT ... FOR UPDATE
            String selectSQL = "SELECT id FROM t WHERE status = false LIMIT 1 FOR UPDATE";
            selectStmt = conn.prepareStatement(selectSQL);
            rs = selectStmt.executeQuery();

            if (rs.next()) {
                int id = rs.getInt("id");

                // 如果查询结果不为空,执行UPDATE操作
                String updateSQL = "UPDATE t SET status = true WHERE id = ?";
                updateStmt = conn.prepareStatement(updateSQL);
                updateStmt.setInt(1, id);
                updateStmt.executeUpdate();

                // 提交事务
                conn.commit();
                System.out.println("Transaction committed. Row updated with id: " + id);
            } else {
                System.out.println("No rows found with status = false.");
                conn.rollback(); // 没有行需要更新,回滚事务
            }
        } catch (SQLException e) {
            e.printStackTrace();
            if (conn != null) {
                try {
                    conn.rollback(); // 出现异常时回滚事务
                } catch (SQLException ex) {
                    ex.printStackTrace();
                }
            }
        } finally {
            // 关闭资源
            try {
                if (rs != null) rs.close();
                if (selectStmt != null) selectStmt.close();
                if (updateStmt != null) updateStmt.close();
                if (conn != null) conn.close();
            } catch (SQLException ex) {
                ex.printStackTrace();
            }
        }
    }
}

关键点

  1. 事务

    • 使用 START TRANSACTION 开启事务。
    • 在更新成功后使用 COMMIT 提交事务;如果没有要更新的行或出现异常,则使用 ROLLBACK 回滚事务。
  2. 行锁

    • 使用 SELECT ... FOR UPDATE 语句锁定选中的行,防止其他事务在当前事务提交前修改这些行。
  3. 处理结果

    • 在编程逻辑中处理查询结果。如果查询返回不为空,则执行更新操作。
  4. 错误处理

    • 在出现异常时,确保回滚事务以保持数据一致性。

通过这种方式,可以确保在高并发环境下,只会有一个事务成功更新满足条件的行,从而避免竞争条件和数据不一致的问题。


二. 那么SELECT ... FOR UPDATE 语句 在MySql 中是加的读锁还是写锁呢?

在MySQL中,SELECT ... FOR UPDATE 语句会为选中的行加上写锁(也称为排他锁或X锁),而不是读锁。写锁可以防止其他事务在当前事务提交之前读取或修改这些行,从而确保数据的一致性和正确性。

写锁的作用

  • 防止并发修改:当一个事务使用SELECT ... FOR UPDATE选中行并加上写锁后,其他事务不能对这些行进行任何更新操作,直到当前事务完成(提交或回滚)。
  • 防止读取:在MySQL的默认InnoDB存储引擎下,SELECT ... FOR UPDATE选中的行也不能被其他事务读取(即使是读锁,也不能加在这些行上),确保当前事务的独占性。

示例

假设有一张名为 orders 的表,包含以下列:idstatusamount

开启事务并加锁

START TRANSACTION;

SELECT id FROM orders WHERE status = 'pending' LIMIT 1 FOR UPDATE;

在上面的SQL语句中,符合条件的行会被加上写锁。其他事务在当前事务完成之前无法修改或读取这些行。

更新选中的行

UPDATE orders SET status = 'completed' WHERE id = <selected_id>;

COMMIT;

举例说明并发控制

假设有两个事务T1和T2同时运行:

事务T1:

START TRANSACTION;

SELECT id FROM orders WHERE status = 'pending' LIMIT 1 FOR UPDATE;  -- 假设选中了id为1的行
-- id 为 1 的行被锁定,其他事务无法读取或修改

UPDATE orders SET status = 'completed' WHERE id = 1;

COMMIT;

事务T2:

START TRANSACTION;

SELECT id FROM orders WHERE status = 'pending' LIMIT 1 FOR UPDATE;  -- 等待T1完成,因为id为1的行被T1锁定

-- 等待T1提交后,选中未被锁定的下一行

UPDATE orders SET status = 'completed' WHERE id = 2;

COMMIT;

在这种情况下,SELECT ... FOR UPDATE 确保了事务T1对选中行的独占访问权。只有在T1提交后,T2才能继续并对数据进行操作,从而避免了数据竞争和不一致问题。

结论

  • SELECT ... FOR UPDATE 在MySQL中为选中的行加上写锁(排他锁)。
  • 写锁确保其他事务在当前事务提交之前无法读取或修改被锁定的行。
  • 这种机制有助于在高并发环境中确保数据的一致性和正确性。
posted @ 2024-07-04 22:13  gongchengship  阅读(27)  评论(0编辑  收藏  举报