Многие из backend-разработчиков получали ошибки с неприятным содержанием, суть которого можно описать двумя словами: deadlock detected. Эти ошибки коварные. Возникают они там, где их не ждёшь, отладочной информации крайне мало или вообще нет, и для их решения необходимо глубокое понимание архитектуры как самого запроса и метода, из которого он вызвался (или методов, возможно, чужих...), так и архитектуры самой СУБД. Поэтому часто у таких ошибок либо переносится срок, либо попытки их исправить приводят к тому, что они возвращаются снова и снова. А deadlock-и так никуда и не исчезают.

Прежде чем говорить о взаимоблокировках отметим, что любое изменение записи — добавление, редактирование и удаление — 1) происходит внутри конкретной транзакции и 2) всегда накладывает на запись блокировку. Для упрощения я не буду здесь рассматривать частичные блокировки — это не сильно влияет на понимание сути проблемы. Просто условимся, что всегда блокируется запись целиком. При этом прочитать запись можно (СУБД-блокировочники в расчёт не берём), и прочитана будет та версия записи, которая доступна текущей транзакции на основании её уровня изоляции и состояния БД в момент её старта. При этом транзакция, которая изменила запись, конечно же увидит свои изменения. Но если другие транзакции попытаются что-либо сделать с этой же записью — отредактировать, удалить или добавить такую же (при наличии уникального индекса), они встанут на наложенную другой транзакцией блокировку и будут ждать, пока изменившая запись транзакция не завершится.

Взаимоблокировка, или deadlock, — это состояние, когда из-за наложившихся друг на друга блокировок дождаться завершения какой-либо из транзакций, удерживающих запись, становится невозможно. К счастью, все СУБД умеют отслеживать взаимоблокировки и принудительно завершать блокирующие транзакции по истечении определённого лимита времени. Это для них как фундамент, основа, без которой СУБД не может существовать. Но наличие подобного механизма не означает, что разработчику ничего не нужно делать с взаимоблокировками — напротив, важно найти причину взаимоблокировки и её устранить. В этой статье я постарался собрать известные сценарии взаимоблокировок, встречающиеся в повседневной практике и рассмотрел их на примере СУБД PostgreSQL. Также отдельно рассмотрен особый вид взаимоблокировок — взаимоблокировки на уровне приложения, с которыми СУБД самостоятельно справиться не может в принципе. Итак, приступим.

Сценарий 1. Классическая взаимоблокировка

Сценарий классической взаимоблокировки следующий:

  1. Стартуют две транзакции — А и Б.

  2. Транзакция А меняет запись 1.

  3. Транзакция Б меняет запись 2.

  4. Транзакция А пытается изменить запись 2 и встаёт на блокировку, ожидая завершения транзакции Б.

  5. Транзакция Б пытается изменять запись 1 и встаёт на блокировку, ожидая завершения транзакции А.

Всё. Транзакции уперлись друг в друга, и единственный способ разрешить эту смертельную блокировку – принудительно завершить одну из них.

Заметьте, что взаимоблокировка возникла из-за того, что транзакции 1 и 2 меняют записи в разном порядке. Если бы порядок был бы одинаковый, взаимоблокировки бы не возникло:

  1. Стартуют две транзакции — А и Б.

  2. Транзакция А меняет запись 1.

  3. Транзакция Б меняет запись 1 и встаёт на блокировку, ожидая завершения транзакции А.

  4. Транзакция А меняет запись 2 (ведь ей уже никто не мешает это сделать).

  5. Транзакция А завершается.

  6. Транзакция Б перечитывает запись 1, изменённую транзакцией А, и в свою очередь тоже меняет её.

  7. Транзакция Б перечитывает запись 2, изменённую транзакцией А, и также меняет её.

  8. Транзакция Б завершается.

Как видно, никаких взаимоблокировок в этом случае не происходит. Отсюда делаем важный вывод: первый и основной способ борьбы с взаимоблокировками — правильный порядок обновления записей.

Сценарий 2. Взаимоблокировка вследствие нескольких запросов

Рассмотрим следующий вид взаимоблокировок, который довольно часто вызывает у разработчиков непонимание. Это происходит, когда внутри транзакции меняется одна запись, но у разных таблиц. Бедный разработчик, получивший такую ошибку, может долго смотреть на запрос, изменяющий одну-единственную запись по ключу, и недоумевать, откуда здесь взаимоблокировка? Я же ОДНУ! запись меняю! Но давайте шире посмотрим на этот пример и приведём сценарий подобной взаимоблокировки:

  1. Стартует метод Foo. В методе запрос редактирует запись А таблицы Документ.

  2. Стартует метод Bar. В методе запрос редактирует запись Б таблицы Работа.

  3. В методе Foo следующий запрос редактирует запись Б таблицы Работа и встаёт на блокировку транзакции, запущенной методом Bar.

  4. В методе Bar следующий запрос редактирует запись А таблицы Документ и встаёт на блокировку транзакции, запущенной методом Foo.

И снова вот она взаимоблокировка. Всё отличие от сценария 1 лишь в том, что запись меняют разные запросы. Но это не имеет никакого значения — запросы ведь выполняются в одной транзакции! А значит здесь подойдёт тот же метод борьбы — использовать один и тот же порядок изменения записей. Только в данном случае это будет порядок выполнения запросов. Вот тут может быть проблема, поскольку методы, изменяющие одни и те же таблицы, могут писать разные разработчики из совершенно разных отделов, и договориться о порядке изменения может быть намного сложнее — придётся найти ответственного за конкурирующий метод разработчика и договориться с ним о порядке изменения записей.

Сценарий 3. Взаимоблокировка при множественных изменениях внутри конкурирующих запросов с CTE

Третий сценарий сложный, и борьба с подобными взаимоблокировками требует понимания как архитектуры самих запросов, так и архитектуры СУБД. Он возникает у тех разработчиков, которые любят большие запросы с множественными изменениями. Перед описанием сценария этого вида взаимоблокировки попытаемся представить себе архитектуру подобных запросов.

Допустим, есть запрос, который одновременно и добавляет новые записи в таблицу, и изменяет существующие. Добавление записей контролируется уникальным индексом. Разработчик написал отдельные CTE — одно для изменения, внутри которого выполняется оператор UPDATE, а другое — для добавления с оператором INSERT. Разработчик грамотный, поэтому всё сделал правильно — учёл порядок записей как в запросе изменения, так и в запросе добавления записей. При этом записи для изменения предварительно заблокировал оператором SELECT с командой FOR (NO KEY) UPDATE. Запрос работает отлично, но... иногда приходят ошибки с взаимоблокировками... между одними и теми же запросами, запущенными разными бизнес-логиками. Что за чертовщина? Всё же учтено!

Всё, да не всё. Тут нужно вспомнить две особенности архитектуры PostgreSQL, касающиеся оптимизации. Первая состоит в том, что оптимизатор PostgreSQL может вообще не выполнять некоторые CTE, если видит, что данные из них нигде не будут использоваться (например, в конце итогового запроса стоит банальный SELECT TRUE). Но CTE, содержащие операторы изменения данных — INSERT, UPDATE и DELETE — выполняются всегда. И вторая — оптимизатор может решить выполнить разные CTE в разных потоках. И какой из них начнёт выполняться раньше, даже сам оптимизатор не знает.

И вот теперь мы готовы представить себе картину происходящего:

  1. Бизнес-логика 1 запускает метод Foo с запросом, изменяющим и добавляющим данные.

  2. Бизнес-логика 2 запускает тот же самый метод Foo с тем же запросом, но с другими данными (частично или полностью пересекающимися с первыми).

  3. Оптимизатор СУБД решает выполнить оба запроса в параллельных потоках.

  4. Стартует CTE добавления данных запроса из бизнес-логики 1 — поток, выполняющий добавление, освободился раньше.

  5. Стартует CTE изменения данных запроса из бизнес-логики 2 — здесь быстрее освободился поток, выполняющий изменение. Пока всё хорошо, данные не пересекаются...

  6. В запросе из бизнес-логики 1 стартует CTE изменения данных и встаёт на блокировку, наложенную транзакцией бизнес-логики 2.

  7. В запросе из бизнес-логики 2 стартует CTE добавления данных и встаёт на блокировку, наложенную транзакцией бизнес-логики 1 (у нас же уникальный индекс!)...

И вот она, взаимоблокировка! Здравствуйте, где не ждали. Но если присмотреться, все признаки возникновения взаимоблокировки налицо - присутствует произвольный порядок выполнения CTE. Только теперь это произошло не по вине разработчика, а вследствие изменения данных в разных параллельных потоках, порядок выполнения которых не регламентирован.

Метод борьбы остался тем же самым — явно указать порядок выполнения CTE. Но как обмануть оптимизатор в этом случае? С помощью принудительной сериализации выполнения CTE. Для этого достаточно в одной CTE использовать результат выполнения другой, к примеру, в CTE с оператором INSERT добавить холостое условие, использующее CTE с оператором UPDATE, скажем, WHERE (SELECT COUNT(1) FROM update_cte) >= 0. Тогда оптимизатор поймёт, что CTE на добавление он сможет выполнить только после CTE на изменение, и не будет распараллеливать их выполнение.

Сценарий 4. Взаимоблокировка на уровне приложения

Но мы же умные, и можем создать взаимоблокировку, с которой СУБД вообще не в силах справиться! Для этого платформа backend-разработки должна предоставлять возможность выполнять разные методы в отдельных транзакциях. Это необходимое условие для создания взаимоблокировки на уровне приложения. С одной стороны, это хорошо, поскольку изолирует изменения, выполняемые методом. Но хорошо только до определённого момента.

Представьте себе следующий сценарий:

  1. Стартует метод Foo. Стартует транзакция А.

  2. Внутри метода Foo выполняется запрос, изменяющий одну запись Rec.

  3. Далее метод Foo вызывает метод Bar (который выполняется в отдельной транзакции).

  4. Метод Bar стартует новую транзакцию Б и выполняет запрос, изменяющий ту же самую запись Rec.

Что произошло? Поскольку запись Rec уже изменена в транзакции А, и транзакция А не завершена, запрос в методе Bar, выполняемый в транзакции Б, встаёт на блокировку транзакции А. А транзакция А не может завершиться, поскольку завершить её должен метод Foo, который в свою очередь, ожидает завершение метода Bar. Мы получили классический deadlock, за тем лишь исключением, что СУБД не знает, что транзакции А и Б принадлежат одному и тому же методу, и, следовательно, не может принудительно завершить одну из них.

Отследить и устранить такое может только служба эксплуатации, когда диагностирует наглухо зависший метод. Но уже очевидно, что такую ситуацию нельзя допускать. Какие для этого есть способы?

  1. Не использовать выполнение метода в отдельных транзакциях без необходимости (заворачивать вложенные методы в общую транзакцию).

  2. По возможности не допускать изменение одной и той же записи в разных методах.

  3. Использовать таймауты на методах, чтобы метод мог быть принудительно завершён хотя бы по таймауту.

Скорее всего, это не все сценарии возникновения взаимоблокировок. Надеюсь, мне удалось донести мысль, что бороться с взаимоблокировками надо с помощью глубокого изучения происходящих процессов как в бизнес-логике, так и в СУБД. Поистине, труд разработчика никогда не будет скучным! Буду рад, если статья окажется полезной, и в результате количество взаимоблокировок в ваших приложениях сократится.

Комментарии (16)


  1. RideBear51
    25.06.2026 08:31

    В sql server вообще можно дедлок устроить 1 простым update без транзакций из-за индексов и их page блокировок, если запросы летят под нагрузкой (p-p, p-u)


    1. imschur Автор
      25.06.2026 08:31

      Интересно, а есть подробный сценарий? Как боретесь?


      1. RideBear51
        25.06.2026 08:31

        У нас раньше пропадала часть обновлений, пока разбираться не стали. Проблема исчезла после пересоздания индексов с ALLOW_PAGE_LOCKS = OFF


  1. Akina
    25.06.2026 08:31

    первый и основной способ борьбы с взаимоблокировками — правильный порядок обновления записей

    ...

    здесь подойдёт тот же метод борьбы — использовать один и тот же порядок изменения записей. Только в данном случае это будет порядок выполнения запросов.

    Интересно, как вы собираетесь управлять порядком обновления записей? SQL всё же декларативен, а потому в общем случае управлять порядком обновления записей мы не можем - ну за исключением случая, когда выполняется итеративное обновление по одной записи в цикле внутри хранимого объекта. К слову, в Постгрессе у UPDATE даже кляузы ORDER BY нету...

    К тому же зачастую "правильный порядок" обеспечить невозможно в принципе, даже в итеративной процедуре - если обновляются основная таблица и связанная, мы можем обеспечить порядок только по одной из таблиц.


    1. imschur Автор
      25.06.2026 08:31

      Обычно для этого пользуемся SELECT...FOR UPDATE, там порядок блокировки можно указать


      1. Akina
        25.06.2026 08:31

        Ваш первый сценарий, насколько я понимаю, предполагает ОДИН запрос на обновление с SELECT-подзапросом/CTE, осуществляющим блокирование, в каждой транзакции. Надо ли понимать вас так, что в этом SELECT присутствует не только FOR UPDATE, но также ORDER BY по уникальному выражению и LIMIT с избыточно высокой границей?

        Во втором сценарии, как я понимаю, речь идёт о многозапросной транзакции, и там вы сперва в один или несколько запросов блокируете с помощью FOR UPDATE все записи, подлежащие изменению, во всех таблицах, и только потом приступаете к фактическому обновлению, так? А не слишком ли тормозно получается суммарно?

        PS. А на каком уровне изоляции вы всё это проделываете?


        1. imschur Автор
          25.06.2026 08:31

          Сценарии схематичные, конечно. В реальности мы перед обновлением или добавлением записей сначала формируем требуемые данные в отдельных CTE, в которых присутствует ORDER BY и FOR UPDATE. И потом уже по предварительно заблокированным записям выполняем UPDATE, либо INSERT в установленном порядке. В результате имеем добавление/изменение с фиксированным порядком записей. Операций несколько больше, зато без взаимоблокировок.

          Используем READ COMMITTED. Это удобный уровень изоляции, потому что если встанем на заблокированную запись, PostgreSQL перед её изменением перечитает её, когда её отпустит конкурирующая транзакция. По большому счёту это соответствует нашим задачам.

          По поводу LIMIT - да, мы стараемся не допускать слишком масштабного изменения записей в одной транзакции. Тут можно не только LIMIT-ом регулировать, но и на уровне БЛ, например, решать, сколько записей на обновление отправлять в запрос. Поэтому есть несколько вариантов.


          1. Akina
            25.06.2026 08:31

            По поводу LIMIT - да, мы стараемся не допускать слишком масштабного изменения записей в одной транзакции. Тут можно не только LIMIT-ом регулировать, но и на уровне БЛ, например, решать, сколько записей на обновление отправлять в запрос. Поэтому есть несколько вариантов.

            Вы меня, наверное, не поняли. LIMIT, причём с параметром, значение которого заведомо больше количества выбираемых записей - чуть ли не критически необходим, хотя по логике получения финального результата выборки он совершенно не имеет смысла. Задача запроса - выбрать и вернуть, не более, а блокирование есть дополнительная операция и чуть ли не побочный результат. Только наличие LIMIT действительно гарантирует установленный в ORDER BY порядок, потому что лишает оптимизатор всякого манёвра и заставляет его работать строго по тексту. Без него ORDER BY проскакивает из CTE либо подзапроса во внешний запрос лишь случайно, а видимость сохранения сортировки, точнее, факт её сохранения, определяется всего лишь физикой накопления и хранения (промежуточного) записей, но не логикой, по которой передача записей во внешний запрос в сортированном порядке лишена смысла. Запрос выполняется с сортировкой, это да, но никаких действий для её сохранения при передаче сервером не предпринимается, оно как бы "само получается". И то, что в середину не вклинивается никакой потусторонний процесс, больше похоже на везение. Которое в определённый момент может и закончиться. Это как порядок вычисления полей выходного набора - раньше всегда было по синтаксису, а теперь лучше не рисковать, чай не Аксесс, где это документировано..


            1. imschur Автор
              25.06.2026 08:31

              Не совсем так. Когда выполняется SELECT...ORDER BY...FOR (NO KEY) UPDATE, он как раз делает то, что нам нужно - сначала отбирает записи в нужном порядке, указанном в ORDER BY, и потом блокирует их в этом порядке. Никакие дополнительные действия вроде LIMIT для этого не требуются. И далее, когда вы будете изменять заблокированные записи, порядок уже будет не важен - ведь записи уже заблокированы в вашей транзакции, только это сделано до UPDATE. И, кстати, если мы хотим зафиксировать результат CTE, можно использовать материализацию.

              Относительно избыточных вычислений - тут нужно рассматривать в комплексе. Если у вас будет дедлок, одна из транзакций будет откачена, а вместе с ней и все изменения, возможно, других запросов, если были. И тогда придётся заново выполнять весь метод со всеми его вычислениями. Если же мы уходим от дедлока ценой некоторых дополнительных вычислений (за всё приходится платить...) то в целом мы оказываемся в выигрыше, плюс повышается стабильность системы в целом


              1. Akina
                25.06.2026 08:31

                он как раз делает то, что нам нужно - сначала отбирает записи в нужном порядке, указанном в ORDER BY, и потом блокирует их в этом порядке.

                Вот у меня как раз большие сомнения насчёт того, что сперва ВСЁ отбирается, и только потом отобранное блокируется. В документации на https://www.postgresql.org/docs/current/sql-select.html#SQL-FOR-UPDATE-SHARE есть такая фраза:

                If a LIMIT is used, locking stops once enough rows have been returned to satisfy the limit (but note that rows skipped over by OFFSET will get locked).

                И я её воспринимаю как "отобрали запись, сверились со счётчиком, если нужно - блокируем и ищем следующую запись, иначе завершаем отбор". Иначе я не могу понять, почему блокируются и записи от начала и до OFFSET. Ведь эти записи не входят в ОТОБРАННЫЙ набор, но входят в ПРОСКАНИРОВАННЫЙ, и если бы блокирование выполнялось ПОСЛЕ отбора, эти начальные записи не должны были бы блокироваться.


                1. imschur Автор
                  25.06.2026 08:31

                  Как раз наоборот. Если бы не работал ORDER BY, заблокировались бы произвольные записи, а не те, что стоят перед OFFSET. Проверьте :)

                  На самом деле, если бы записи сначала блокировались, а потом сортировались, это было бы серьёзным нарушением логики работы запроса. То есть запрос не выполнял бы команды, которые в нём указаны. Это критическая ошибка так-то, и сообщество никогда не выпустило бы базу с такой ошибкой на прод. Поэтому используйте FOR (NO KEY) UPDATE смело и не переживайте :)

                  К слову - у нас годами работают запросы, построенные на подобной логике, и ни одного дедлока. А раньше были, до того как её внедрили.


                  1. Akina
                    25.06.2026 08:31

                    При чём тут сортировка? Она выполняется безусловно, я же не о ней веду речь-то. Я о том, что блокирование выполняется после сортировки, но в процессе отбора, а не после него. А по завершении отбора, когда записи заблокированы - всё, порядок блокирования утрачен. Настала пора отдавать итоговый результат, с отобранными записями и списком блокировок, наружу. И, поскольку это подзапрос, то отдача записей из него должна обеспечить полноту отобранного набора, но не порядок, который обеспечил внутренний ORDER BY. То есть блокирование записей выполнено в установленном нами порядке, но вот совпадение этого порядка с тем порядком, в котором они будут обрабатываться во внешнем запросе - этой гарантии я не вижу, за исключением физического порядка размещения ссылок в оперативной памяти, чему гарантии я тоже не вижу. То есть мы опираемся не на логику, а на особенность реализации. А это ненадёжно, хотя и работает. Пока что работает..

                    Впрочем, для первого случая это, наверное, уже неважно. Нам важен порядок блокирования, а не порядок высвобождения.


                    1. imschur Автор
                      25.06.2026 08:31

                      Так нам и нужно, чтобы записи блокировались в определённом порядке. В каком порядке потом мы их будем обновлять - уже не важно, ведь они уже заблокированы нами. То есть это решение проблемы дедлоков. А если говорить о том, в каком порядке записи должны отдаваться наружу - то да, тут итоговый ORDER BY нужен. Но, кажется, это уже другая тема


                  1. Akina
                    25.06.2026 08:31

                    А это отдельно.

                    Если бы не работал ORDER BY, заблокировались бы произвольные записи, а не те, что стоят перед OFFSET. Проверьте :)

                    Интересно, совпадает ли список излишне (до OFFSET) заблокированных записей в случаях:

                    FROM ( SELECT t.*
                           FROM t
                           ORDER BY ..
                           LIMIT .. OFFSET ..
                           FOR UPDATE )

                    и

                    FROM ( SELECT t1.*
                           FROM t t1
                           JOIN ( SELECT id
                                  FROM t
                                  ORDER BY ..
                                  LIMIT .. OFFSET .. ) t2 USING (id)
                           FOR UPDATE )

                    Или во втором случае избытка не будет?


                    1. imschur Автор
                      25.06.2026 08:31

                      Полагаю, вы правы, во втором случае заблокируются ровно те записи, которые нужны. Но надо проверить, как оптимизатор строит план по такому кейсу. Как время будет, гляну


            1. imschur Автор
              25.06.2026 08:31

              Вдогонку. Видимо, вы переживаете, что оптимизатор перестроит план запроса так, что в результате записи будут заблокированы в другом порядке, а не в том, что указан в ORDER BY. Такого не произойдёт. FOR (NO KEY) UPDATE - команда, через которую оптимизатор "перепрыгнуть" не может, и в результате план запроса с FOR UPDATE может отличаться от запроса без него. Что нам и требуется