而是触发器或计算列?哪个更好?

时间:2012-03-28 16:31:15

标签: sql sql-server-2008 triggers calculated-columns

我想知道以下两种方法之间是否有任何性能差异。 基本上,问题是我们允许在id中使用空格和破折号,但某些遗留应用程序无法使用这些,因此它们被删除。 据我所知,最好的方法是在触发器中或作为计算列。 SQL如下所示(清理并匿名,如果出现错误则道歉) 到目前为止,在我们的测试服务器上,两种方法之间似乎没有任何区别,其他人是否有任何输入?

[数据库SQL Server 2008] [查找表20000000行并且不断增长]

选项1 - 创建触发器

CREATE TRIGGER triMem_Lkup on Mem_Lkup
INSTEAD OF INSERT
AS
BEGIN
  INSERT INTO Mem_lkup
       SELECT ex_id, contact_gid, id_type_code, date_time_created,
              (replace(replace([ex_id],' ',''),'-','')) as ex_id_calc
       FROM inserted
END
GO

对战 选项2 - 使用计算列

CREATE TABLE [dbo].[Mem_lkup](
    [mem_lkup_sid] [int] IDENTITY(1,1) NOT NULL,
    [ex_id] [varchar](18) NOT NULL,
    [contact_gid] [int] NOT NULL,
    [id_type_code] [char] (1) NOT NULL,
    [date_time_created] [datetime] NOT NULL,
    [ex_id_calc]  AS CAST( replace( replace([ex_id],' ','')  ,'-','')  AS varchar(18)) PERSISTED

    CONSTRAINT [PK_Mem_Lkup] PRIMARY KEY NONCLUSTERED 
(
    [mem_lkup_sid] ASC
)

哪一个最好?

1 个答案:

答案 0 :(得分:6)

计算列最好。

INSTEAD OF触发器将首先在inserted中创建整个伪tempdb表。

Plan

对于带有CREATE TABLE语句的触发器版本(堆上的非群集PK)

SET STATISTICS IO ON;

INSERT INTO [_test].[dbo].[Mem_lkup]
           ([ex_id]
           ,[contact_gid]
           ,[id_type_code]
           ,[date_time_created])
SELECT type AS  [ex_id]
      ,1 [contact_gid]
      ,'A' [id_type_code]
      ,getdate() [date_time_created]
  FROM master..spt_values

给我

Table 'Worktable'. Scan count 0, logical reads 5076
Table 'spt_values'. Scan count 1, logical reads 15

Table 'Mem_lkup'. Scan count 0, logical reads 7549
Table 'Worktable'. Scan count 1, logical reads 15

虽然计算的列版本相似,但避免了worktable读取。

Table 'Mem_lkup'. Scan count 0, logical reads 7555
Table 'spt_values'. Scan count 1, logical reads 15

你有什么理由坚持这个价值吗? (与非持久计算列相反)