Учитывая это:
create table z
(
i int identity(1,1) not null,
zzz int not null
);
При попытке выполнить следующее ..
begin try
begin transaction
alter table z drop column aaa;
commit tran;
end try
begin catch
print 'hello';
SELECT
ERROR_NUMBER() as ErrorNumber,
ERROR_MESSAGE() as ErrorMessage;
IF (XACT_STATE()) = -1
BEGIN
PRINT
N'The transaction is in an uncommittable state. ' +
'Rolling back transaction.'
ROLLBACK TRANSACTION;
END;
end catch
print 'reached';
.. ошибка может быть поймана:
ErrorNumber ErrorMessage
4924 ALTER TABLE DROP COLUMN failed because column 'aaa' does not exist in table 'z'.
Но попробуйте изменить alter table z drop column aaa;
на alter table z add zzz int;
, Sql Server может перехватить ошибку ..
Имена столбцов в каждой таблице должны быть уникальными. Название столбца 'zzz' в таблице
'z' указано более одного раза.
.. но вам не удастся вернуть управление, блок CATCH не сработает. Кажется, нет жестких и быстрых правил, какие ошибки можно отследить, а какие нет.
Чтобы проиллюстрировать разницу, вот код, который можно отследить по вашему коду
data:image/s3,"s3://crabby-images/ac967/ac967bb313004bc2b22817544aa7385559753893" alt="This is fine, the control is yielded back to you, you can perform other things when you CATCH the error"
Вот ошибка, которую ваш код не может отловить, что похоже на вашу проблему.
data:image/s3,"s3://crabby-images/f4669/f4669c09b546c95e089af8b3236169791edd6d2f" alt="The error was caught by Sql Server but won't yield back the control to you. Consequently, you cannot perform other things, your CATCH section will be rendered useless"
Обратите внимание, что там нет сетки (через SELECT ERROR_NUMBER() as ErrorNumber, ERROR_MESSAGE() as ErrorMessage;
). Это означает, что Sql Server не вернул вам элемент управления после обнаружения исключения.
Возможно, здесь вы найдете другие подробности, которые могут помочь: http://msdn.microsoft.com/en-us/library/ms179296.aspx
См. Данное руководство по обработке ошибок @@ ERROR и / или TRY - CATCH
<Ч />
Кстати, в Postgresql все виды ошибок DDL распознаются вашим кодом.
do $$
begin
-- alter table z drop column aaa;
alter table z add zzz int;
exception when others then
raise notice 'The transaction is in an uncommittable state. '
'Transaction was rolled back';
raise notice 'Yo this is good! --> % %', SQLERRM, SQLSTATE;
end;
$$ language 'plpgsql';
Вот сообщение об ошибке от разработчика для alter table z drop column aaa;
в Postgresql:
data:image/s3,"s3://crabby-images/74a36/74a36aaa20a2dca1a0d76c3e900ab630f3c64e6d" alt="enter image description here"
Вот сообщение об ошибке от разработчика для alter table z add zzz int;
в Postgresql; который, кстати, в Sql Server, когда он имеет ошибку в операторе такого типа, не вернет вам управление, поэтому ваши CATCH
разделы иногда полезны, иногда бесполезны.
data:image/s3,"s3://crabby-images/920d7/920d7a7ed1d3156ef450363c8fa1b49f05ad8305" alt="in Sql Server, control execution won't be yielded back to you when there's an exception on adding of column"