169 похожих чатов

День добрый Как произвести "безопасную" операцию с SQL таблицей? К примеру есть

стандартная форма регистрации с email и паролем, на сервер отправляется запрос на создание нового пользователя с заданными данными, но перед этим в базе проверяется не зарегистрирован ли уже пользователь с таким email (в данном примере email уникальное значение строки)

Вопрос. Если параллелью несколько людей попытаются зарегистрироваться с одинаковым email адресом не произойдет ли такое что все они успешно пройдут регистрацию?
Ведь на момент их регистрации email адрес еще не будет добавлен в таблицу.

Возможно я не совсем понимаю как SQL реагирует на параллельные запросы к базе данных, буду признателен если кто-нибудь поведает как это происходит, или даст хорошее чтиво.

7 ответов

48 просмотров

https://www.php.net/manual/ru/book.pdo.php

В SQL есть транзакции и уровни изоляции этих транзакций. Стоит об этом почитать. А так, если очень кратко: по умолчанию включëн автокоммит: это значит каждый запрос будет выполняться в своей неявной транзакции. Вместо этого необходимо сделать так: 1) запустить транзакцию 2) проверить наличие строки с помощью SELECT ... FOR UPDATE 3) если еë нет, вставить такую строку SELECT ... FOR UPDATE повесит пессимистическую блокировку, что будет держать остальные запросы на эту же строку.

allpeg
В SQL есть транзакции и уровни изоляции этих транз...

блокировать надо таблицу, а не строку. Пока вы блокируете несуществующую строку, другие запросы из очереди добавят этот email. Разве нет?

Vitaliy Monashenkov
блокировать надо таблицу, а не строку. Пока вы бло...

Блокируется не несуществующая строка, а все ключи из запроса

allpeg
Блокируется не несуществующая строка, а все ключи ...

какие ключи? SELECT… FOR UPDATE — блокирует считываемые строки на чтение. Точно такую же блокировку ставит обычный UPDATE, когда считывает данные для обновления.

allpeg
В SQL есть транзакции и уровни изоляции этих транз...

Разве нельзя уровень изоляции поднять до serializable?

@Thermophoto
Разве нельзя уровень изоляции поднять до serializa...

При уровне serializable если сделать запрос на select а затем insert с двух транзакций будет одно из двух (в зависимости от того какая бд) 1) запрос на select станет блокирующим другую транзакцию 2) оба запроса на селект скажут что всё ок, но потом один из INSERT ов выкинет ошибку сериализации

Похожие вопросы

Обсуждают сегодня

30500 за редактор? )
Владимир
47
Недавно Google Project Zero нашёл багу в SQLite с помощью LLM, о чём достаточно было шумно в определённых интернетах, которые сопровождались рассказами, что скоро всех "ибешни...
Alex Sherbakov
5
вы делали что-то подобное и как? может есть либы готовые? увидел картинку нокода, где всё линиями соединено и стало интересно попробовать то же в ddl на lua сделать. решил с ч...
Victor
8
Подскажите пожалуйста, как в CustomDrawCell(Sender: TcxCustomGridTableView; ACanvas: TcxCanvas; AViewInfo: TcxGridTableDataCellViewInfo; var ADone: Boolean); получить наз...
A Z
7
Ребят в СИ можно реализовать ООП?
Николай
33
https://github.com/erlang/otp/blob/OTP-27.1/lib/kernel/src/logger_h_common.erl#L174 https://github.com/erlang/otp/blob/OTP-27.1/lib/kernel/src/logger_olp.erl#L76 15 лет назад...
Maksim Lapshin
20
Как передать управляющий символ в открытую через CreateProcess консоль? Собсна, есть процедура: procedure TRedirectThread.WriteData(Data: OEMString); var Written: Cardinal;...
Serjone
1
Он в одиночку это дело запилил или была какая-то команда?
Aquinary
12
~ 2m21s  nix shell github:nixos/nixpkgs#stack ~  stack ghc -- --version error: … while calling the 'derivationStrict' builtin at /builtin/derivation.nix:...
Rebuild your mind.
6
Всем привет, нужна как никогда, нужна помощь с IO в загрузчике. Пишу в code16 после установки сегментных регистров, пишу вывод символа. Пробовал 2 варианта: # 1 mov $0x0E, %a...
Shadow Akira
14
Карта сайта