MSSQLSERVER_4104

适用范围:SQL Server

详细信息

Attribute
产品名称 SQL Server
事件 ID 4104
事件来源 MSSQLSERVER
组件 SQLEngine
符号名称 ALG_MULTI_ID_BAD
消息正文 无法绑定由多个部分组成的标识符 "%.*ls"。

说明

SQL Server 中实体的名称称为其 标识符。 在引用实体(例如在查询中指定列名和表名)时可以使用标识符。 由多个部分组成的标识符包含一个或多个作为标识符前缀的限定符。 例如,表标识符可能以诸如包含表的数据库和架构的名称之类的限定符作为前缀,而列标识符可能以诸如表名称或表别名之类的限定符作为前缀。

错误 4104 表示无法将指定的由多个部分组成的标识符映射到现有实体。 在以下情况下可能会返回此错误:

  • 作为列名称前缀提供的限定符与查询中使用的任何表名或别名均不对应。

    例如,下面的语句使用表别名 (Dept) 作为列前缀,但是此表别名未在 FROM 子句中引用。

    SELECT Dept.Name FROM HumanResources.Department;  
    

    在下面的语句中,在属于两个表之间的 JOIN 条件的 WHERE 子句中指定了由多个部分组成的列标识符 TableB.KeyCol,但是 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 列的属性是通过在列名称和属性名称之间使用句号分隔符 (.) 引用的,而这与用表名称作为列名称前缀的方式相同。 以下示例创建两个表:ab。 表 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
数据库标识符