Oracle 12c - 插入到选择查询中的不明确列,ORA-00918

Posted

技术标签:

【中文标题】Oracle 12c - 插入到选择查询中的不明确列,ORA-00918【英文标题】:Oracle 12c - Ambiguous column in Insert Into Select Query, ORA-00918 【发布时间】:2020-02-07 08:33:36 【问题描述】:

我正在尝试使用单个语句执行多个插入来实现这一点,我正在使用 Insert into select 语句。但是当两列在插入中具有相同的值时,我面临着。我收到的错误消息是 ORA-00918: column ambiguously defined

查询

INSERT INTO sample (
    HOST,
    TOTAL_PING,
    TOTAL_UNAVAILABLE_PING
)

SELECT * FROM (

    SELECT 'FR3158-73-1',
    82,
    82
    FROM DUAL
    UNION ALL

    SELECT 'FR3158-76-2',
    80,
    10
    FROM DUAL
)

如果我将一个值更改为有效的值,则在第一个选择语句中存在两个值是 82 和 82 的问题。即使列值相同,我也不知道如何进行这项工作。

--- 更新 ---

表定义

CREATE TABLE sample
(
  ID                      NUMBER GENERATED ALWAYS as IDENTITY(START with 1 INCREMENT by 1) PRIMARY KEY,
  HOST                    VARCHAR2(15 BYTE),
  TOTAL_PING              INTEGER,
  TOTAL_UNAVAILABLE_PING  INTEGER,
  ADDED_ON                TIMESTAMP(6)          DEFAULT systimestamp
);

【问题讨论】:

你能包含sample的表定义吗? @TimBiegeleisen 问题已更新,我已添加表定义 【参考方案1】:

在这种情况下,您不需要子查询 - 正如@Littlefoot 所示。但是如果你这样做了,在更复杂的情况下,你可以通过在子查询中为列表达式设置别名来避免错误:

INSERT INTO sample (
    HOST,
    TOTAL_PING,
    TOTAL_UNAVAILABLE_PING
)

SELECT * FROM (

    SELECT 'FR3158-73-1' as host,
    82 as total_ping,
    82 as total_unavailable_ping 
    FROM DUAL
    UNION ALL

    SELECT 'FR3158-76-2',
    80,
    10
    FROM DUAL
)
/

2 rows inserted.

问题在于子查询本身会获得隐含的列别名,这些别名源自查询的第一个分支中的值:

SELECT 'FR3158-73-1',
82,
82 
FROM DUAL
UNION ALL

SELECT 'FR3158-76-2',
80,
10
FROM DUAL

'FR3158-73-         82         82
----------- ---------- ----------
FR3158-73-1         82         82
FR3158-76-2         80         10

第二列和第三列都称为"82",这是ORA-00918 抱怨的歧义,来自外部select。如果添加消失的别名:

SELECT 'FR3158-73-1' as host,
82 as total_ping,
82 as total_unavailable_ping 
FROM DUAL
UNION ALL

SELECT 'FR3158-76-2',
80,
10
FROM DUAL

HOST        TOTAL_PING TOTAL_UNAVAILABLE_PING
----------- ---------- ----------------------
FR3158-73-1         82                     82
FR3158-76-2         80                     10

所以外部查询不再混淆。请注意,您只需要联合的第一个分支中的别名 (usually, anyway) - 将它们放在所有分支中并没有什么坏处,它们只会被忽略,但如果您是手动创建它。在这种情况下,实际的别名也无关紧要,它们必须是唯一的;具体来说,它们不必与您要插入的列相匹配 - 但如果它们匹配,则更容易跟踪。

如果您按照@Littlefoot 显示的那样进行操作,则表明您没有中间结果集select,因此不需要评估派生名称(如果可以说它们完全存在的话),所以歧义看不到 - 它纯粹是位置。

【讨论】:

谢谢亚历克斯..你已经详细解释了完整的场景..支持:)【参考方案2】:

删除select * from((以及尾随的))。

INSERT INTO sample (HOST, TOTAL_PING, TOTAL_UNAVAILABLE_PING)
   SELECT 'FR3158-73-1', 82, 82 FROM DUAL
           UNION ALL
           SELECT 'FR3158-76-2', 80, 10 FROM DUAL

[编辑,在评论后它仍然不起作用]

嗯,它工作,至少在我的 11gXE 中:

SQL> select * From v$version where rownum = 1;

BANNER
--------------------------------------------------------------------------------
Oracle Database 11g Express Edition Release 11.2.0.2.0 - 64bit Production

SQL> CREATE TABLE sample
  2  (
  3    ID                      NUMBER,
  4    HOST                    VARCHAR2(15 BYTE),
  5    TOTAL_PING              INTEGER,
  6    TOTAL_UNAVAILABLE_PING  INTEGER,
  7    ADDED_ON                TIMESTAMP(6)          DEFAULT systimestamp
  8  );

Table created.

SQL> INSERT INTO sample (HOST, TOTAL_PING, TOTAL_UNAVAILABLE_PING)
  2     SELECT 'FR3158-73-1', 82, 82 FROM DUAL
  3             UNION ALL
  4             SELECT 'FR3158-76-2', 80, 10 FROM DUAL;

2 rows created.

12c 也没有错误:

SQL> select * from v$version where rownum = 1;

BANNER                                                                               CON_ID
-------------------------------------------------------------------------------- ----------
Oracle Database 12c Enterprise Edition Release 12.2.0.1.0 - 64bit Production              0

SQL> CREATE TABLE sample
  2  (
  3    ID                      NUMBER,
  4    HOST                    VARCHAR2(15 BYTE),
  5    TOTAL_PING              INTEGER,
  6    TOTAL_UNAVAILABLE_PING  INTEGER,
  7    ADDED_ON                TIMESTAMP(6)          DEFAULT systimestamp
  8  );

Table created.

SQL> INSERT INTO sample (HOST, TOTAL_PING, TOTAL_UNAVAILABLE_PING)
  2     SELECT 'FR3158-73-1', 82, 82 FROM DUAL
  3             UNION ALL
  4             SELECT 'FR3158-76-2', 80, 10 FROM DUAL;

2 rows created.

SQL>

现在,请证明它不适用于您的数据库。

【讨论】:

不错的一个 +1。你知道一些方法来修复 OP 正在使用的确切代码吗? Alex 在他的回答中做到了这一点,@Tim。 Ummm .. 什么是“它”,显示相同的错误? 您的查询显示的错误与我的问题中提到的相同。 我通过在实时数据库、11gXE 和 12cR2 上添加示例来编辑我的答案 - 它们都没有错误。【参考方案3】:

尝试命名列:

    INSERT INTO sample (HOST, TOTAL_PING, TOTAL_UNAVAILABLE_PING)
       SELECT 'FR3158-73-1' as host, 82 as total_ping, 82 as total_UNAVAILABLE_PING FROM DUAL
          UNION ALL
       SELECT 'FR3158-76-2' as host, 80 as total_ping, 10 as total_UNAVAILABLE_PING FROM DUAL
;

但更好的方法是:

    INSERT INTO sample (HOST, TOTAL_PING, TOTAL_UNAVAILABLE_PING) values ('FR3158-73-1',82,82);
    INSERT INTO sample (HOST, TOTAL_PING, TOTAL_UNAVAILABLE_PING) values ('FR3158-76-2',80,10);

你避免子查询。 如果您有大量数据要插入数据库,请考虑使用 SQL*Loader 并从文本文件加载。 https://docs.oracle.com/en/database/oracle/oracle-database/12.2/sutil/oracle-sql-loader-concepts.html#GUID-DD843EE2-1FAB-4E72-A115-21D97A501ECC

【讨论】:

以上是关于Oracle 12c - 插入到选择查询中的不明确列,ORA-00918的主要内容,如果未能解决你的问题,请参考以下文章

从集合类型 oracle 12c 插入表 - ORA-00902:无效数据类型

Oracle中的多行插入查询(从一张表中选择多行并插入到另一张表中[重复]

Oracle 12c 中的子选择性能不佳

Oracle 12c - 使用另一个表中的值在表中插入值

将 SQL 查询转换为 PL/SQL 可以提高 Oracle 12c 中的性能吗? [关闭]

Oracle 12c 子查询的 WITH 子句中的函数