SET XACT_ABORT 的用法

本文介绍了SQL Server中SETXACT_ABORT特性的使用方法及作用。当开启此特性时,若Transact-SQL语句执行期间出现运行时错误,则整个事务会自动回滚;关闭时仅回滚引发错误的语句。文章还提供了示例代码来演示不同设置下的行为差异。

SET XACT_ABORT

指定当 Transact-SQL 语句产生运行时错误时,Microsoft® SQL Server™ 是否自动回滚当前事务。

语法

SET XACT_ABORT { ON | OFF }

注释

SET XACT_ABORTON 时,如果 Transact-SQL 语句产生运行时错误,整个事务将终止并回滚。为 OFF 时,只回滚产生错误的 Transact-SQL 语句,而事务将继续进行处理。编译错误(如语法错误)不受 SET XACT_ABORT 的影响。

对于大多数 OLE DB 提供程序(包括 SQL Server),隐性或显式事务中的数据修改语句必须将 XACT_ABORT 设置为 ON。唯一不需要该选项的情况是提供程序支持嵌套事务时。有关更多信息,请参见分布式查询和分布式事务。

SET XACT_ABORT 的设置是在执行或运行时设置,而不是在分析时设置。

示例

下例导致在含有其它 Transact-SQL 语句的事务中发生违反外键错误。在第一个语句集中产生错误,但其它语句均成功执行且事务成功提交。在第二个语句集中,SET XACT_ABORT 设置为 ON。这导致语句错误使批处理终止,并使事务回滚。

 

CREATE   TABLE  t1 (a  int   PRIMARY   KEY )
CREATE   TABLE  t2 (a  int   REFERENCES  t1(a))
GO
INSERT   INTO  t1  VALUES  ( 1 )
INSERT   INTO  t1  VALUES  ( 3 )
INSERT   INTO  t1  VALUES  ( 4 )
INSERT   INTO  t1  VALUES  ( 6 )
GO
SET  XACT_ABORT  OFF
GO
BEGIN   TRAN
INSERT   INTO  t2  VALUES  ( 1 )
INSERT   INTO  t2  VALUES  ( 2 /* Foreign key error */
INSERT   INTO  t2  VALUES  ( 3 )
COMMIT   TRAN
GO

SET  XACT_ABORT  ON
GO

BEGIN   TRAN
INSERT   INTO  t2  VALUES  ( 4 )
INSERT   INTO  t2  VALUES  ( 5 /* Foreign key error */
INSERT   INTO  t2  VALUES  ( 6 )
COMMIT   TRAN
GO

/* Select shows only keys 1 and 3 added. 
   Key 2 insert failed and was rolled back, but
   XACT_ABORT was OFF and rest of transaction
   succeeded.
   Key 5 insert error with XACT_ABORT ON caused
   all of the second transaction to roll back. 
*/


SELECT   *  
FROM  t2
GO

DROP   TABLE  t2
DROP   TABLE  t1
GO

 

USE [DXQcontrol_L1HIMCP1_Operational_W] GO /****** Object: StoredProcedure [dbo].[prodExecmgmtIdent__processBR] Script Date: 7/29/2025 9:03:21 AM ******/ SET ANSI_NULLS ON GO SET QUOTED_IDENTIFIER ON GO /*----------------------------------------------------------------------- Project: Duerr EcoEmos - Module Ident Stored Procedure: dbo.prodExecmgmtIdent__processBR Author: JM Initial date: 11. Nov 2020 Revisions: rev 1 - 11. Nov 2020, JM: wrapper function for the old OrderManagement to call dbo.prodExecmgmtIdent__processDR Input: @arg_dateEvt PLC timestamp of the reading event @arg_szRWStationTag Tagname for the reader (configured in table DS_CfgRegistrationPoint_TAB and for FB369 in addition in file IdentServer.ini) @arg_szDatasetIn Tracking data from PLC Output: @arg_szDatasetOut The mds data string which was merged by the new data from PLC and the last stored data string from DS_MDS_TAB Return: See the return codes of the several variables with name @iRetCode... ------------------------------------------------------------------------*/ CREATE PROCEDURE [dbo].[prodExecmgmtIdent__processBR] @arg_dateEvt DATETIME ,@arg_szRWStationTag NStringMin ,@arg_szDatasetIn VARCHAR(MAX) ,@arg_szDatasetOut VARCHAR(MAX) OUTPUT AS BEGIN SET NOCOUNT ON; SET XACT_ABORT ON; SET DEADLOCK_PRIORITY HIGH; DECLARE @object_name VARCHAR(128) = OBJECT_SCHEMA_NAME(@@PROCID) + '.' + OBJECT_NAME(@@PROCID); ---------------------------------------------------------------------------------------------------- -- PRREQ-1908: avoid start of process with an open or uncommitted transaction DECLARE @log VARCHAR(MAX) = ''; DECLARE @tempTable TABLE(logExt VARCHAR(MAX)); IF XACT_STATE() <> 0 BEGIN SET @log = 'Start ' + @object_name + '; '; SET @log += '@@TRANCOUNT = ' + CAST(@@TRANCOUNT AS VARCHAR(MAX)) + '; '; SET @log += 'XACT_STATE() = ' + CAST(XACT_STATE() AS VARCHAR(MAX)) + '; '; SET @log += '@@spid = ' + CAST(@@spid AS VARCHAR(MAX)) + '; '; SET @log += 'ProcParam: '; SET @log += 'rwStation: ' + @arg_szRWStationTag + '; '; SET @log += 'bodyId: ' + dbo.DS_FCT_GetPLCValueOf_(@arg_szDatasetIn, 'bodyId') + '; '; SET @log += 'skidId: ' + dbo.DS_FCT_GetPLCValueOf_(@arg_szDatasetIn, 'skidId') + '; '; SET @log += 'skidType: ' + dbo.DS_FCT_GetPLCValueOf_(@arg_szDatasetIn, 'skidType') + '; '; INSERT INTO @tempTable (logExt) SELECT DISTINCT 'spid=' + CAST(S.spid AS VARCHAR(MAX)) + '; hostname=' + RTRIM(S.hostname) + '; program_name=' + RTRIM(S.program_name) + '; hostprocess=' + RTRIM(S.hostprocess) + '; text=' + D.text + '; lastwaittype=' + RTRIM(S.lastwaittype) FROM sys.sysprocesses S CROSS APPLY sys.dm_exec_sql_text(S.sql_handle) D WHERE S.open_tran > 0; --ORDER BY S.hostname, S.program_name; ROLLBACK TRANSACTION; INSERT INTO dbo.DS_MDSErrorLog_TAB (dateEvt, rwStationId, source, errorCode, errorText) SELECT GETDATE(), @arg_szRWStationTag, @object_name, 'WARNING', isnull(logExt,'-') FROM @tempTable; EXEC dbo.DS_SP_MDSWriteToErrorLog @arg_rwStationId = @arg_szRWStationTag, @arg_szSource = @object_name, @arg_szErrorCode = 'WARNING', @arg_szErrorText = @log; END; ---------------------------------------------------------------------------------------------------- DECLARE @iRetValue INT; ------------------------------------------------------------------------------- -- write this information into log EXEC [dbo].[DS_SP_MDSWriteToLogger] @arg_szRWStationTag, @object_name, @arg_szDatasetIn; ------------------------------------------------------------------------------- -- set the prober function code SET @arg_szDatasetIn = STUFF( @arg_szDatasetIn, 1, 2, 'DR'); EXEC @iRetValue = [dbo].[prodExecmgmtIdent__processDR] @arg_dateEvt, -- DATETIME @arg_szRWStationTag, -- NStringMin @arg_szDatasetIn, -- VARCHAR(MAX) @arg_szDatasetOut OUTPUT; -- VARCHAR(MAX) RETURN @iRetValue; END; GO 逐条解析sqlserverSql语句语法用法
07-30
USE [irepodb] GO /****** Object: StoredProcedure [dbo].[P_12#_BOLT_GG] Script Date: 2025/8/29 10:57:11 ******/ SET ANSI_NULLS ON GO SET QUOTED_IDENTIFIER ON GO ALTER PROCEDURE [dbo].[P_12#_BOLT_GG] @def_top_org VARCHAR(10) IN ('27942','26697','31581','16992') -- 参数化默认值 AS BEGIN SET NOCOUNT ON; SET XACT_ABORT ON; -- 错误时自动回滚事务 BEGIN TRY -- 1. 备份并清理旧数据 BEGIN TRANSACTION; INSERT INTO T_BOLT_TREND_B SELECT * FROM T_BOLT_TREND WHERE def_top_id = @def_top_org; DELETE FROM T_BOLT_TREND WHERE def_top_id = @def_top_org; -- 2. 使用CTE准备数据 ;WITH BoltData AS ( SELECT b.displayValue, b.celladdress, b.rep_top_id, b.def_top_id, b.sheetNo, b.def_top_org, -- 解析行列号 col_num = dbo.getTransCOL(REPLACE(SUBSTRING(b.celladdress, 2, 2), '$', '')), row_num = CONVERT(INT, REPLACE(RIGHT(RTRIM(b.celladdress), 2), '$', '')) FROM dbo.T_BOLT b WHERE b.def_top_org = @def_top_org AND b.type IN ('InputNumeric', 'KeyboardText') AND CONVERT(INT, REPLACE(RIGHT(RTRIM(b.celladdress), 2), '$', '')) >= 6 AND b.displayValue IS NOT NULL AND (dbo.getTransCOL(REPLACE(SUBSTRING(b.celladdress, 2, 2), '$', '')) - 71) % 3 != 0 ), MetaData AS ( SELECT rep_top_id, sheetNo, MAX(CASE WHEN celladdress = '$H$2:$I$2' THEN displayValue END) AS [Date], MAX(CASE WHEN celladdress = '$H$3:$I$3' THEN displayValue END) AS Workpiece FROM dbo.T_BOLT WHERE def_top_org = @def_top_org GROUP BY rep_top_id, sheetNo ) -- 3. 批量插入趋势数据(含时间列修复) INSERT INTO T_BOLT_TREND ( ID, Date, TIME, Workpiece, Differ, DisplayValue, Orders, COL, Specs, updateTime, rep_top_id, def_top_id, sheetNo, Remarks ) SELECT NEWID(), md.[Date], COALESCE( -- 精确时间定位 ( SELECT TOP 1 t.displayValue FROM dbo.T_BOLT t WHERE t.rep_top_id = bd.rep_top_id AND t.sheetNo = bd.sheetNo AND t.type = 'Time' AND dbo.getTransCOL(REPLACE(SUBSTRING(t.celladdress, 2, 2), '$', '')) = CASE WHEN (bd.col_num - 71) % 3 = 1 THEN bd.col_num + 1 WHEN (bd.col_num - 71) % 3 = 2 THEN bd.col_num END AND CONVERT(INT, REPLACE(RIGHT(RTRIM(t.celladdress), 2), '$', '')) = 4 ), '' -- 最终默认值 ) AS TIME, md.Workpiece, CASE WHEN (bd.col_num - 71) % 3 = 1 THEN '最大值' WHEN (bd.col_num - 71) % 3 = 2 THEN '最小值' END AS Differ, bd.displayValue, CASE WHEN (bd.col_num - 71) % 3 = 1 THEN (bd.col_num - 69) / 3 ELSE (bd.col_num - 70) / 3 END AS Orders, REPLACE(bd.celladdress, ' ', ''), COALESCE( ( SELECT TOP 1 displayValue FROM dbo.T_BOLT spec WHERE spec.rep_top_id = bd.rep_top_id AND spec.sheetNo = bd.sheetNo AND spec.def_top_org = bd.def_top_org AND dbo.getTransCOL(REPLACE(SUBSTRING(spec.celladdress, 2, 2), '$', '')) = CASE WHEN (bd.col_num - 71) % 3 = 1 THEN bd.col_num + 2 ELSE bd.col_num + 1 END ), 'N/A' -- 规格默认值 ) AS Specs, GETDATE(), bd.rep_top_id, bd.def_top_org, bd.sheetNo, '12' FROM BoltData bd JOIN MetaData md ON bd.rep_top_id = md.rep_top_id AND bd.sheetNo = md.sheetNo; -- 4. 插入看板数据 INSERT INTO T_BOLT_KANBAN SELECT A.Specs, A.DATE, A.TIME, A.Workpiece, A.Differ, A.COL, CONVERT(DECIMAL(12, 2), A.DisplayValue) AS DisplayValue, A.Orders, A.rep_top_id, A.def_top_id, A.sheetNo, B.PROJECTID, B.PROJECT, B.ROW, B.BOLT_NUM, B.LINE, B.TEAM, B.TYPE, B.UP_LIMIT, B.DOWN_LIMIT, B.UP_ACTION, B.DOWN_ACTION, GETDATE() FROM dbo.T_BOLT_TREND A LEFT JOIN dbo.T_BOLT_LIMIT B ON A.def_top_id = B.DEF_TOP_ORG AND B.ROW = REPLACE(RIGHT(RTRIM(A.COL), 2), '$', '') WHERE A.def_top_id = @def_top_org; COMMIT TRANSACTION; -- 5. 数据质量验证 DECLARE @EmptyTimeCount INT; SELECT @EmptyTimeCount = COUNT(*) FROM T_BOLT_KANBAN WHERE def_top_id = @def_top_org AND TIME = ''; END TRY BEGIN CATCH IF @@TRANCOUNT > 0 ROLLBACK TRANSACTION; RAISERROR('存储过程执行错误.', 16, 1); END CATCH END
08-30
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值