Does “select for update” prevent other connections inserting when the row is not present

前端 未结 5 1031
轮回少年
轮回少年 2020-12-30 21:24

I\'m interested in whether a select for update query will lock a non-existent row.

e.g.

Table FooBar with two columns, foo and bar, foo has a un

相关标签:
5条回答
  • 2020-12-30 21:55

    In Oracle, the SELECT ... FOR UPDATE has no effect on a non-existent row (the statement simply raises a No Data Found exception). The INSERT statement will prevent a duplicates of unique/primary key values. Any other transactions attempting to insert the same key values will block until the first transaction commits (at which time the blocked transaction will get a duplicate key error) or rolls back (at which time the blocked transaction continues).

    0 讨论(0)
  • 2020-12-30 21:59

    I wrote a detailed analysis of this thing on SQL Server: Developing Modifications that Survive Concurrency

    Anyway, you need to use SERIALIZABLE isolation level, and you really need to stress test.

    0 讨论(0)
  • 2020-12-30 22:03

    SQL Server only has the FOR UPDATE as part of a cursor. And, it only applies to UPDATE statements that are associated with the current row in the cursor.

    So, the FOR UPDATE has no relationship with INSERT. Therefore, I think your answer is that it's not applicable in SQL Server.

    Now, it may be possible to simulate the FOR UPDATE behavior with transactions and locking strategies. But, that may be more than what you're looking for.

    0 讨论(0)
  • 2020-12-30 22:16

    MySQL

    SELECT ... FOR UPDATE with UPDATE

    Using transactions with InnoDB (auto-commit turned off), a SELECT ... FOR UPDATE allows one session to temporarily lock down a particular record (or records) so that no other session can update it. Then, within the same transaction, the session can actually perform an UPDATE on the same record and commit or roll back the transaction. This would allow you to lock down the record so no other session could update it while perhaps you do some other business logic.

    This is accomplished with locking. InnoDB utilizes indexes for locking records, so locking an existing record seems easy--simply lock the index for that record.

    SELECT ... FOR UPDATE with INSERT

    However, to use SELECT ... FOR UPDATE with INSERT, how do you lock an index for a record that doesn't exist yet? If you are using the default isolation level of REPEATABLE READ, InnoDB will also utilize gap locks. As long as you know the id (or even range of ids) to lock, then InnoDB can lock the gap so no other record can be inserted in that gap until we're done with it.

    If your id column were an auto-increment column, then SELECT ... FOR UPDATE with INSERT INTO would be problematic because you wouldn't know what the new id was until you inserted it. However, since you know the id that you wish to insert, SELECT ... FOR UPDATE with INSERT will work.

    CAVEAT

    On the default isolation level, SELECT ... FOR UPDATE on a non-existent record does not block other transactions. So, if two transactions both do a SELECT ... FOR UPDATE on the same non-existent index record, they'll both get the lock, and neither transaction will be able to update the record. In fact, if they try, a deadlock will be detected.

    Therefore, if you don't want to deal with a deadlock, you might just do the following:

    INSERT INTO ...

    Start a transaction, and perform the INSERT. Do your business logic, and either commit or rollback the transaction. As soon as you do the INSERT on the non-existent record index on the first transaction, all other transactions will block if they attempt to INSERT a record with the same unique index. If the second transaction attempts to insert a record with the same index after the first transaction commits the insert, then it will get a "duplicate key" error. Handle accordingly.

    SELECT ... LOCK IN SHARE MODE

    If you select with LOCK IN SHARE MODE before the INSERT, if a previous transaction has inserted that record but hasn't committed yet, the SELECT ... LOCK IN SHARE MODE will block until the previous transaction has completed.

    So to reduce the chance of duplicate key errors, especially if you hold the locks for awhile while performing business logic before committing them or rolling them back:

    1. SELECT bar FROM FooBar WHERE foo = ? LOCK FOR UPDATE
    2. If no records returned, then
    3. INSERT INTO FooBar (foo, bar) VALUES (?, ?)
    0 讨论(0)
  • 2020-12-30 22:17

    On Oracle:

    Session 1

    create table t (id number);
    alter table t add constraint pk primary key(id);
    
    SELECT *
    FROM t
    WHERE id = 1
    FOR UPDATE;
    -- 0 rows returned
    -- this creates row level lock on table, preventing others from locking table in exclusive mode
    

    Session 2

    SELECT *
    FROM t 
    FOR UPDATE;
    -- 0 rows returned
    -- there are no problems with locking here
    
    rollback; -- releases lock
    
    
    INSERT INTO t
    VALUES (1);
    -- 1 row inserted without problems
    
    0 讨论(0)
提交回复
热议问题