表值构造函数 (Transact-SQL)
指定要构建到某一表中的一组行值表达式。Transact-SQL 表值构造函数允许在单个 DML 语句中指定多行数据。该表值构造函数可以在 INSERT 语句的 VALUES 子句中指定,在 MERGE 语句的 USING <源表> 子句中指定,以及在 FROM 子句的派生表定义中指定。
语法
VALUES ( <row value expression list> ) [ ,...n ]
<row value expression list> ::=
{<row value expression> } [ ,...n ]
<row value expression> ::=
{ DEFAULT | NULL | expression }
参数
VALUES
介绍行值表达式列表。每个列表都必须用括号括起来并由逗号分隔。在每个列表中指定的值的数目必须相同,并且值必须采用与表中的列相同的顺序。表中每个列的值必须指定,或者列列表必须显式为每个传入值指定列。
DEFAULT
强制数据库引擎插入为列定义的默认值。如果某列并不存在默认值,并且该列允许 Null 值,则插入 NULL。DEFAULT 对标识列无效。当在表值构造函数中指定时,只在 INSERT 语句中允许 DEFAULT。expression
一个常量、变量或表达式。表达式不能包含 EXECUTE 语句。
限制和局限
可以使用表值构造函数构造的最大行数为 1000。若要插入超过 1000 行的数据,请创建多个 INSERT 语句,或通过使用 bcp 实用工具或 BULK INSERT 语句大容量导入数据。
只允许单个标量值作为行值表达式。涉及多列的子查询不允许作为行值表达式。例如,以下代码导致语法错误,因为第三个行值表达式列表包含具有多列的子查询。
USE AdventureWorks;
GO
CREATE TABLE dbo.MyProducts (Name varchar(50), ListPrice money);
GO
-- This statement fails because the third values list contains multiple columns in the subquery.
INSERT INTO dbo.MyProducts (Name, ListPrice)
VALUES ('Helmet', 25.50),
('Wheel', 30.00),
(SELECT Name, ListPrice FROM Production.Product WHERE ProductID = 720);
GO
但是,可以通过单独在子查询中指定每一列,重新编写该语句。下面的示例成功地将三行插入 MyProducts 表中。
INSERT INTO dbo.MyProducts (Name, ListPrice)
VALUES ('Helmet', 25.50),
('Wheel', 30.00),
((SELECT Name FROM Production.Product WHERE ProductID = 720),
(SELECT ListPrice FROM Production.Product WHERE ProductID = 720));
GO
数据类型
在多行 INSERT 语句中指定的值遵循 UNION ALL 语法的数据类型约定属性。这导致不匹配类型隐式转换到更高优先级的类型。如果此转换不是所支持的隐式转换,则返回错误。例如,以下语句将整数值和字符值插入到类型为 char 的列中。
CREATE TABLE dbo.t (a int, b char);
GO
INSERT INTO dbo.t VALUES (1,'a'), (2, 1);
GO
运行 INSERT 语句时,SQL Server 尝试将 'a' 转换为整数,因为数据类型优先级指示整数类型的优先级高于字符。转换失败,并且返回错误。您可以根据需要显式转换值,从而避免发生此错误。例如,前面的语句可以编写为:
INSERT INTO dbo.t VALUES (1,'a'), (2, CONVERT(CHAR,1));
示例
A. 插入多行数据
下面的示例创建表 dbo.Departments,然后使用表值构造函数将五行数据插入到该表中。由于提供了所有列的值并按表中各列的顺序列出这些值,因此不必在列列表中指定列名。
USE AdventureWorks;
GO
INSERT INTO Production.UnitMeasure
VALUES (N'FT2', N'Square Feet ', '20080923'), (N'Y', N'Yards', '20080923'), (N'Y3', N'Cubic Yards', '20080923');
GO
B. 使用 DEFAULT 和 NULL 值插入多行
下面的示例说明如何在使用表值构造函数向表中插入行时指定 DEFAULT 和 NULL。
USE AdventureWorks;
GO
CREATE TABLE Sales.MySalesReason(
SalesReasonID int IDENTITY(1,1) NOT NULL,
Name dbo.Name NULL ,
ReasonType dbo.Name NOT NULL DEFAULT 'Not Applicable' );
GO
INSERT INTO Sales.MySalesReason
VALUES ('Recommendation','Other'), ('Advertisement', DEFAULT), (NULL, 'Promotion');
SELECT * FROM Sales.MySalesReason;
C. 在 FROM 子句中将多个值指定为派生表
下面的示例在 SELECT 语句的 FROM 子句中使用表值构造函数指定多个值。
SELECT a, b FROM (VALUES (1, 2), (3, 4), (5, 6), (7, 8), (9, 10) ) AS MyTable(a, b);
GO
D. 在 MERGE 语句中将多个值指定为派生源表
下面的示例使用 MERGE 以更新或插入行的方式来修改 SalesReason 表。当源表中的 NewName 值与目标表 (SalesReason) 的 Name 列中的值匹配时,就会更新此目标表中的 ReasonType 列。当 NewName 的值不匹配时,就会将源行插入到目标表中。此源表是一个派生表,它使用 Transact-SQL 表值构造函数指定源表的多个行。
USE AdventureWorks;
GO
-- Create a temporary table variable to hold the output actions.
DECLARE @SummaryOfChanges TABLE(Change VARCHAR(20));
MERGE INTO Sales.SalesReason AS Target
USING (VALUES ('Recommendation','Other'), ('Review', 'Marketing'), ('Internet', 'Promotion'))
AS Source (NewName, NewReasonType)
ON Target.Name = Source.NewName
WHEN MATCHED THEN
UPDATE SET ReasonType = Source.NewReasonType
WHEN NOT MATCHED BY TARGET THEN
INSERT (Name, ReasonType) VALUES (NewName, NewReasonType)
OUTPUT $action INTO @SummaryOfChanges;
-- Query the results of the table variable.
SELECT Change, COUNT(*) AS CountPerChange
FROM @SummaryOfChanges
GROUP BY Change;