Существуют некоторые практические проблемы с отключением триггера в производственной среде. Я предполагаю, что вы будете специально отключать триггеры таблиц (в отличие от базы данных или всего сервера):
- Вам нужны разрешения ALTER для таблицы, на которой вы работаете. Это обычно считается проблематичным по соображениям безопасности. Это становится намного более проблематичным на уровне базы данных и сервера.
- Отключение не ограничено вашим конкретным подключением / сеансом, но повлияет на все события, которые запускают триггер, из любых сеансов, пока он отключен. Если вы ожидаете параллелизма в рассматриваемом коде, вы должны предположить, что тот же код будет вызываться из других потоков. Чего не хватает в предыдущих предложениях, так это полного рассмотрения управления параллелизмом, которого требует использование любого семафора. В частности, эта временная таблица определяет семафор для защищенного раздела кода (в смысле многопоточности), где семафор - это существование временной таблицы. Если вы на самом деле отключаете триггер, то для правильного использования этой временной таблицы по прямому назначению необходимо провести тест непосредственно перед тем, как защищенный код создаст временную таблицу, чтобы проверить, существует ли она уже. Для этого может потребоваться глобальная временная таблица, или вы можете выполнить более умный поиск в базе данных tempdb, чтобы увидеть, существует ли она в каком-либо сеансе. Защищенный код должен проверить, существует ли семафор, и войти в состояние блокировки, если он намеревается дождаться, когда ресурс (в данном случае триггер) будет доступен, как и ожидалось. Это становится сложным, и имеет небольшую ценность в обмен на сложность.
Мнение. Несмотря на то, что триггеры легко отключить / включить, есть много соображений, которые следует соблюдать осторожно, если вы временно отключите их как часть какого-то конкретного варианта использования, поскольку это разработка стратегии управления параллелизмом.
Если вы не достаточно комфортно разбираетесь в управлении параллелизмом или если у вас нет абсолютно никаких требований к тому, чтобы код был реентерабельным / параллельным, у вас (вероятно, спорадически) будут проблемы, если вы используете DISABLE TRIGGER, но не не учитывать эти факторы.
Самый безопасный путь, который я вижу, учитывая все это, - это не отключать триггер, использовать локальную временную таблицу в качестве семафора, который влияет только на текущий сеанс, тщательно кодировать код выхода, чтобы временная таблица была определенно уничтожена при конец защищенного кода, как уже предлагалось, но он все еще требует проверки / блокировки, даже если это будет иметь значение только при одном и том же соединении и, в частности, при параллельном выполнении на одном и том же соединении. Абсолютно безопасный способ сделать это - создать sproc только для защищенного кода. Sproc проверяет / блокирует, затем, если он продолжается (и вам нужно будет проверить наличие тупиковой ошибки после выхода из кода блокировки), создает временную таблицу. Так как временные значения уничтожаются при возврате sproc, любой путь из защищенного кода будет обрабатывать семафор. Но временные таблицы доступны в течение сеанса - не только внутри sproc (пока sproc работает), либо даже внутри пакета. SQL Server поддерживает параллельные запросы для одного сеанса, поэтому временная таблица, созданная в одном потоке сеанса, видна в любом другом. Это означает, что он может быть замечен вне sproc в том же сеансе, и фактически, тот же самый код может быть запущен в это время. Вот почему в этом сценарии вам по-прежнему необходимо реальное управление параллелизмом.
И, наконец, мои извинения за запутанный комментарий. Я считаю, что почти каждое обсуждение многопоточности и управления параллелизмом превращается в это, потому что, хотя концепции не так уж и сложны, методы кодирования долгое время считались деликатными и хрупкими и подвержены ошибкам разработчика.