详细信息
| 产品名称 | SQL Server |
| 事件编号 | 4104 |
| 事件源 | MSSQLSERVER |
| 组件 | SQLEngine |
| 符号名称 | ALG_MULTI_ID_BAD |
| 消息正文 | 无法绑定多部分标识符“%.*ls”。 |
说明
SQL Server 中实体的名称称为其 标识符。 每当引用实体时,可以使用标识符,例如,通过在查询中指定列名和表名。 多部分标识符包含一个或多个限定符作为标识符的前缀。 例如,表标识符可能带有限定符(例如包含表的数据库名称和架构名称),或者列标识符可能带有限定符(如表名或表别名)的前缀。
错误 4104 指示指定的多部分标识符无法映射到现有实体。 此错误可以在以下情况下返回:
作为列名前缀提供的限定符与查询中使用的任何表或别名不对应。
例如,以下语句使用表别名 (
Dept) 作为列前缀,但 FROM 子句中未引用表别名。SELECT Dept.Name FROM HumanResources.Department;在以下语句中,在 WHERE 子句中指定了多部分列标识符
TableB.KeyCol,作为两个表之间的 JOIN 条件的一部分,TableB但在查询中未显式引用。DELETE FROM TableA WHERE TableA.KeyCol = TableB.KeyCol;SELECT 'X' FROM TableA WHERE TableB.KeyCol = TableA.KeyCol;FROM 子句中提供了表的别名,但为列提供的限定符是表名。 例如,以下语句使用表名
Department作为列前缀;但是,该表在 FROM 子句中引用了别名(Dept)。SELECT Department.Name FROM HumanResources.Department AS Dept;使用别名时,表名不能在语句中的其他位置使用。
SQL Server 无法确定多部分标识符是否引用以表为前缀的列,还是引用以列为前缀的 CLR 用户定义数据类型(UDT)的属性。 发生这种情况是因为 UDT 列的属性通过使用列名和属性名称之间的句点分隔符(.)来引用,方式与列名称以表名为前缀的方式相同。 以下示例创建两个表,
a以及b。 表b包含列,该列a使用 CLR UDTdbo.myudt2作为其数据类型。 SELECT 语句包含多部分标识符a.c2。CREATE TABLE a (c2 int); GOCREATE TABLE b (a dbo.myudt2); GOSELECT a.c2 FROM a, b;假设 UDT 没有命名
c2的属性,SQL Server 无法确定标识符a.c2是引用表中a的列,还是引用表中b的列c2a、属性c2。myudt2
用户操作
将列前缀与查询的 FROM 子句中指定的表名或别名匹配。 如果为 FROM 子句中的表名定义了别名,则只能将该别名用作与该表关联的列的限定符。
可以按如下所示更正引用
HumanResources.Department表的上述语句:SELECT Dept.Name FROM HumanResources.Department AS Dept; GOSELECT Department.Name FROM HumanResources.Department; GO确保查询中指定了所有表,并正确指定了表之间的 JOIN 条件。 可以按如下所示更正上述 DELETE 语句:
DELETE FROM dbo.TableA WHERE TableA.KeyCol = (SELECT TableB.KeyCol FROM TableB WHERE TableA.KeyCol = TableB.KeyCol); GO可以按如下所示更正上述
TableASELECT 语句:SELECT 'X' FROM TableA, TableB WHERE TableB.KeyCol = TableA.KeyCol;或
SELECT 'X' FROM TableA INNER JOIN TableB ON TableB.KeyCol = TableA.KeyCol;对标识符使用唯一明确定义的名称。 这样做可使代码更易于读取和维护,同时最大程度地减少对多个实体的不明确引用的风险。