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

Вечер добрый. Как бороться с race condition? В несколько потоков

клиентский код пытается заселектить, обновить или вставить данные. В таблице уникальный индекс на эти данные. При многопоточном доступе и select for update проскакивает вставка одних и тех же данных. Естественно падает по ограничению целостности. Можно как-то заблокироваться, но не на всей таблице?

8 ответов

7 просмотров

Может это нормально?

Viktor-AB Автор вопроса
Sergey Gr
Может это нормально?

Нужно заблокироваться на строке и не селектить, пока другая транзакция не отпустит.

Вариант 1. Использовать везде SERIALIZABLE (прочитав в документации, как именно), и получить надёжное решение проблемы. Вариант 2. Прочитать и осознать (попробовать на примерах) всю документацию про изоляцию в PostgreSQL (про serializable можно при этом не читать), понять механизмы, подумать о возможных взаимодействиях в конкретном случае, придумать и протестировать подход... и, если Вы хоть что-то сделаете неправильно, то проблема просто будет появляться значительно реже. ;) Для варианта 2 есть набор паттернов, которые часто помогают, а иногда — нет, если что.

Непонятно, в чем проблема? Вы делаете select for update, так? Это значит, что запрос заблокируется, если часть строк, которые вы выбираете, взяты другим select for update. Поэтому тут 2 варианта: 1) вы выбираете разные строки двумя транзакциями, в которых делается select for update, и изменяете их на одни и те же строки, поэтому получаете ошибку 2) либо просто делаете select for update и изменяете данные на те, которые уже вставлены. Выглядит так, что проблема либо в логике приложения, либо в том, что вы не хотите обрабатывать ошибки.

Viktor-AB Автор вопроса
Aleksey Stavrov
Непонятно, в чем проблема? Вы делаете select for u...

Не совсем. Для примера пытаюсь в несколько потоков в одной транзакции сделать select for update и если нет строки, то вставить. Ожидал, что будет работать блокировка, но по факту проскакивают попытки вставить одно и то же в разных транзакциях.

Viktor AB
Не совсем. Для примера пытаюсь в несколько потоков...

А unique constraint над суррогатным ключем или над чем-то существенным?

Viktor-AB Автор вопроса
Sergey Gr
А unique constraint над суррогатным ключем или над...

В синтетическом примере на все поля, кроме ключа.

Viktor AB
Не совсем. Для примера пытаюсь в несколько потоков...

Можете попробовать уровень изоляции serializable поставить, но его нужно будет включать вообще для всех транзакции. Тогда у вас эта ошибка скорей всего уйдёт и будет другая ошибка - ошибка сериализации данных. Более детально тут невозможно что-то подсказать, не поняв логику работы.

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

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

Всем привет! Имеется функция: function IsValidChar(ch: UTF8Char): Boolean; var i: Integer; ValidChars: AnsiString; begin ValidChars := 'abcdefghijklmnopqrstuvwxyzABCDE...
Евгений
44
И никого не интересует какие пакеты кто использует. ((% Заходишь на сайт симфони и видишь поддержку Украины - по законам РФ это ж экстремизм. Только никто не отказывается от с...
Am Ambrion
11
лучше скажите, причём тут паскаль?
Alexey Kulakov
36
Чтобы перехватить все нажимания буков на форме, надо хук ставить? Пробовал на форме ОнКейДаун, оно ловит клаву если фокус не на компоненте с вводом текста
Serjone
15
Но, может, есть уже проверенная? Наши требования такие: 1. Сообщения должны приходить из Инста в CRM оду 2. Должна быть возможность подключить несколько экаунтов Инстаграм. Р...
Alexander Sharoiko MSE / Александр Шаройко
7
Народ! Впервые клиенту пришло письмо от РКН, у вас, дескать, есть яндекс метрика, а нигде не написано, что вы ее юзаете. Никто не сталкивался?
Sasha Beep
14
Всем привет! вывожу на общей стр дочерние ресурсыв каждом ресурсе галерея, и первая фотка должна выводиться на общей [!DocLister? &prepare=photo !]
Alekso
12
Я правильно понимаю что нет способов получить список ожидающих заявок на вступление в группу с помощью бота из mtproto?
Шамиль Прилов
7
А можно вопрос? Мне сегодня сказали что у меня функция (которая просто заполняет массив значениями) не правильная void Full(double * arr, int n) { for (int i = 0; i < n; i...
† C E †
7
Добрый вечер. Хочу чтобы у меня в классе поле было функцией, которая возвращает строку. Делаю так: interface ... TGetOutPath = function : String of object; ... protec...
Kirill Filippenok
12
Карта сайта