MSSQLSERVER_4104

详细信息

产品名称 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 UDT dbo.myudt2 作为其数据类型。 SELECT 语句包含多部分标识符 a.c2

    CREATE TABLE a (c2 int);   
    GO  
    
    CREATE TABLE b (a dbo.myudt2);   
    GO  
    
    SELECT a.c2 FROM a, b;   
    

    假设 UDT 没有命名c2的属性,SQL Server 无法确定标识符a.c2是引用表中a的列,还是引用表中b的列c2a、属性c2myudt2

用户操作

  • 将列前缀与查询的 FROM 子句中指定的表名或别名匹配。 如果为 FROM 子句中的表名定义了别名,则只能将该别名用作与该表关联的列的限定符。

    可以按如下所示更正引用 HumanResources.Department 表的上述语句:

    SELECT Dept.Name FROM HumanResources.Department AS Dept;  
    GO  
    
    SELECT 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  
    

    可以按如下所示更正上述 TableA SELECT 语句:

    SELECT 'X' FROM TableA, TableB WHERE TableB.KeyCol = TableA.KeyCol;  
    

    SELECT 'X' FROM TableA INNER JOIN TableB ON TableB.KeyCol = TableA.KeyCol;  
    
  • 对标识符使用唯一明确定义的名称。 这样做可使代码更易于读取和维护,同时最大程度地减少对多个实体的不明确引用的风险。

另请参阅

MSSQLSERVER_107
数据库标识符