FIX: A MERGE 语句可能不强制外键约束,该语句更新一个唯一的键列不是聚集键的一部分,并且没有时单个行与更新源 SQL Server 2008 年

文章翻译 文章翻译
文章编号: 956718 - 查看本文应用于的产品
错误 #: 50003167 (SQL 修补程序)
生成 SQL 发布之后发布的主列表的详细信息,请单击下面的文章编号,以查看 Microsoft 知识库中相应的文章:
957826您可以在其中找到有关 SQL Server 2008 的详细信息生成 SQL Server 2008 以后发布并 SQL Server 2005 生成 SQL Server 2005 Service Pack 2 以后发布
展开全部 | 关闭全部

症状

在 Microsoft SQL Server 2008 年外, 键约束可能会不强制满足以下条件为真时:
  • 发出 MERGE 语句。
  • 更新的目标列都有一个唯一的非聚集索引。
请考虑以下情形。 该语句更新名为 Column1 被命名为 Table1 的表的唯一列。Table1 从名为 Table2 的表的外键约束被引用。

结果是他们不应该已被时,会更改在 Table1 中的行。此外,Table2 必须具有对 Table1 的虚引用的行。

当满足下列条件都为真时,在此方案中发生此问题:
  • 被引用的 Column1 列,在 Table1 中不是聚集键的 Table1 的一部分。
  • 只有一个可能的值可以被分配给 Column1 列。 例如对于一种在以下情况下发生:
    • 合并源是数据的单个行。 例如对于合并源是从下面选择语句之一:
      • select <ConstantValues>
      • select <Parameters>
      注意这种情况下是最可能的方案。
    • 合并源是实际数据的单个行。 例如对于合并源是从下面选择语句之一:
      • select <ColumnName> from <TableName> where <TableName>.<ColumnName> = 1
        注意<TableName><ColumnName> 称为查询优化器是一个唯一的值。
      • select top 1 <ColumnName> from <TableName>
    • 合并源和合并目标之间联接的保证单个行将会更新该谓词。
    • 更新子句将 Column1 列设置为无需考虑的合并源的常量值。
  • 外键约束中 Table2 上未启用 更新层叠 选项。
注意我们建议您应用此修补程序,如果您使用 MERGE 语句更新具有非聚集的唯一索引引用的外键约束的列。

解决方案

累积更新 1 中第一次释放此问题的修复程序。有关如何获取 SQL Server 2008 年此累积更新包的详细信息单击下面的文章编号,以查看 Microsoft 知识库中相应的文章:
956717累积更新包 1 SQL Server 2008 年
注意因为这些版本是累积性的因此每个新的修补程序版本包含的所有修补程序和所有安全修复程序以前 SQL Server 2008 中包含的修补都程序版本。我们建议您考虑应用最新的修补程序版本包含此修补程序。有关详细的信息请单击下面的文章编号,以查看 Microsoft 知识库中相应的文章:
956909SQL Server 2008 生成 SQL Server 2008 发布之后发布的

替代方法

此修补程序包消除了该问题。如果您在"症状"部分中所描述的情况下使用 MERGE 语句,并选择不应用此修复程序请按照下列步骤来消除此问题:
  1. 重写 MERGE 语句,以便的合并源值是表、 临时的表或表变量,而不是被排中的列在查询中。
  2. 使用跟踪标记 8790。此跟踪标记强制优化程序使用一种称为大范围更新计划的计划。大范围更新计划没有问题。此步骤中都带有性能风险的所有 DML 语句。因此,您应避免使用此步骤,除非它是不可能更改应用程序。
下面的 TRANSACT-SQL 脚本说明更改您的脚本来解决此问题,如果您不能应用此修补程序的一种方法。

例如对于您有一个脚本,它类似于以下内容:
use tempdb;

drop table sale, product;
create table product(pno int not null primary key, name char(30), pAlternateKey char(6) not null unique);
create table sale(sno int not null primary key, pAlternateKey char(6) not null references product(pAlternateKey));
insert product values(1, 'Office Chair', 'ochair');
insert sale values(1, 'ochair')

-- No violation of foreign key constraint is detected. However, one should be.
merge into product
using (select 'Office Chair2' as name, 1 as pno, 'oxx' as pAlternateKey) as src
on product.pno = src.pno
when matched then
   update set product.pAlternateKey = src.pAlternateKey, 
              product.name = src.name
when not matched then
   insert values(src.pno, src.name, src.pAlternateKey);
更改脚本,以使它类似于以下:
insert product values(1, 'Office Chair', 'ochair');
insert sale values(1, 'ochair')
-- A foreign key constraint violation is detected, and the update fails.
declare @source table 
   (name nchar(30), pno int, pAlternateKey nchar(30));
insert into @source values('Office Chair2',1,'oxx');

merge into product
using @source as src
on product.pno = src.pno
when matched then
   update set product.pAlternateKey = src.pAlternateKey, 
              product.name = src.name
when not matched then
   insert values(src.pno, src.name, src.pAlternateKey);

状态

Microsoft 已经确认这是在"适用于"一节中列出的 Microsoft 产品中的问题。

更多信息

有关哪些文件发生更改的详细信息,并应用累积更新包包含此 Microsoft 知识库文章中描述的修补程序的任何系统必备项有关的信息,请单击下面的文章编号,以查看 Microsoft 知识库中相应的文章:
956717累积更新包 1 SQL Server 2008 年

参考

SQL Server 2008 年发布后可用的生成的列表的详细信息,请单击下面的文章编号,以查看 Microsoft 知识库中相应的文章:
956909SQL Server 2008 生成 SQL Server 2008 发布之后发布的


对于 SQL Server 增量的服务模型的详细信息,请单击下面的文章编号,以查看 Microsoft 知识库中相应的文章:
935897一个增量的服务模型是可从 SQL Server 团队提供报告的问题的修补程序


有关命名 SQL Server 更新架构的详细信息,请单击下面的文章编号,以查看 Microsoft 知识库中相应的文章:
822499Microsoft SQL Server 软件更新程序包的新命名架构


有关软件更新术语的详细信息,请单击下面的文章编号,以查看 Microsoft 知识库中相应的文章:
824684用于描述 Microsoft 软件更新的标准术语的说明

参考

有关 SQL Server 2008 中的非聚集索引的详细信息,请访问下面的 Microsoft 开发人员网络 (MSDN) 的网站:
http://msdn.microsoft.com/en-us/library/ms179325(SQL.100).aspx

属性

文章编号: 956718 - 最后修改: 2008年9月19日 - 修订: 4.0
这篇文章中的信息适用于:
  • Microsoft SQL Server 2008 Enterprise
  • Microsoft SQL Server 2008 Developer
  • Microsoft SQL Server 2008 Standard
  • Microsoft SQL Server 2008 Standard Edition for Small Business
  • Microsoft SQL Server 2008 Web
  • Microsoft SQL Server 2008 Workgroup
  • Microsoft SQL Server 2008 Express
  • Microsoft SQL Server 2008 Express with Advanced Services
关键字:?
kbmt kbautohotfix kbhotfixserver kbqfe kbpubtypekc kbfix KB956718 KbMtzh
机器翻译
注意:这篇文章是由无人工介入的微软自动的机器翻译软件翻译完成。微软很高兴能同时提供给您由人工翻译的和由机器翻译的文章, 以使您能使用您的语言访问所有的知识库文章。然而由机器翻译的文章并不总是完美的。它可能存在词汇,语法或文法的问题,就像是一个外国人在说中文时总是可能犯这样的错误。虽然我们经常升级机器翻译软件以提高翻译质量,但是我们不保证机器翻译的正确度,也不对由于内容的误译或者客户对它的错误使用所引起的任何直接的, 或间接的可能的问题负责。
点击这里察看该文章的英文版: 956718
Microsoft和/或其各供应商对于为任何目的而在本服务器上发布的文件及有关图形所含信息的适用性,不作任何声明。 所有该等文件及有关图形均"依样"提供,而不带任何性质的保证。Microsoft和/或其各供应商特此声明,对所有与该等信息有关的保证和条件不负任何责任,该等保证和条件包括关于适销性、符合特定用途、所有权和非侵权的所有默示保证和条件。在任何情况下,在由于使用或运行本服务器上的信息所引起的或与该等使用或运行有关的诉讼中,Microsoft和/或其各供应商就因丧失使用、数据或利润所导致的任何特别的、间接的、衍生性的损害或任何因使用而丧失所导致的之损害、数据或利润不负任何责任。

提供反馈

 

Contact us for more help

Contact us for more help
Connect with Answer Desk for expert help.
Get more support from smallbusiness.support.microsoft.com