Не удаляйте временные таблицы, умоляю

от автора

Мне часто приходится видеть чужой код на T-SQL. Я уже привык видеть в конце процедур привычное

drop table #a drop table #b

Таблица сама себя не удалит, видимо думает автор. Ну или это делает «на всякий случай». Вдруг SQL server забудет ее удалить? Впрочем, эти удаления не столь страшны, так как SQL server их оптимизирует (не делает перекомпиляции, о чем мы поговорим дальше, так как удаление происходит строго в конце). Кроме того, если отладка тела процедуры производится скриптом, то удаления нужны, и когда в самом конце этот скрипт оборачивают в процедуру, то удаления остаются.

Но недавно я стал встречать совершенно жуткий антипаттерн. Не знаю, откуда он распространился.

Встречайте:

  if object_id('tempdb..#mytemp') is not null     DROP TABLE #mytemp   create table #mytemp (...)

Мне обидно, что SQL server считают идиотом со стекающими от вырождения слюнями, идиотом, неспособным заботиться о контексте выполнения. Но важнее то, что это код — потенциальная бомба с часовым механизмом.

Покажем это на примере. Создадим внешнюю процедуру:

create procedure ALPHA as   create table #mytemp (n int, ALPHA varchar(128))   insert into #mytemp select 1, 'ALPHA'   select 1 as point, * from #mytemp   exec BETA   select 2 as point, * from #mytemp GO

Как вы видите, этот код вызывает внутреннюю процедуру BETA:

create procedure BETA as   create table #mytemp (n int, BETA varchar(128))   insert into #mytemp select 1, 'BETA'   select 3 as point, * from #mytemp GO 

Выполним процедуру ALPHA и увидим, что каждая процедура видит объекты в своей области видимости. Временные таблицы также доступны, если они не экранированы созданными локально:

две таблицы сосуществуют вместе, в чем можно убедиться добавив оператор — ***

create procedure BETA as   create table #mytemp (n int, BETA varchar(128))   insert into #mytemp select 1, 'BETA'   select 3 as point, * from #mytemp   select * from tempdb.dbo.sysobjects where name like '%mytemp%' -- *** GO

Вот они, две наши таблички мирно сосуществуют. Мы можем усложнить задачу SQL так:

Я привел скриншот, чтобы обратить внимание на то, что редактор подозревает, что тут ошибка: таблица #mytemp используется после удаления. Но мы знаем, что делаем:

В 3-й отладочной печати выводится локальная таблица, а в 4-й — внешняя, из ALPHA. После drop SQL server вынужден перекомпилировать хвост процедуры, потому что у другой таблицы могут быть другие поля, как в данном случае.

Теперь вас не должно удивить, что произойдет при использовании антипаттерна:

create procedure BETA as   if object_id('tempdb..#mytemp') is not null     DROP TABLE #mytemp     create table #mytemp (n int, BETA varchar(128))   insert into #mytemp select 1, 'BETA'   select 3 as point, * from #mytemp GO
Проверьте себя

Так как проверка и удаление таблицы производится вначале, то удаляется внешняя таблица, созданная процедурой ALPHA. Дальше все идет хорошо до момента, пока ALPHA не пытается прочитать из уже несуществующей таблицы.

Если временные таблицы в процедурах ALPHA и BETA называются по-разному, то все будет хорошо. Все будет хорошо до первого случайного пересечения имен.


ссылка на оригинал статьи https://habr.com/ru/post/723278/


Комментарии

Добавить комментарий

Ваш адрес email не будет опубликован. Обязательные поля помечены *