【问题标题】:SQL Update with subsubquery带有子查询的 SQL 更新
【发布时间】:2011-09-21 13:05:53
【问题描述】:

我在 Oracle 11g 中遇到以下更新查询问题:

update TABLE_A a set COL1 = 
  (SELECT b.COL2 FROM
    (SELECT ROWNUM AS ROW_NUMBER, b.COL2 from TABLE_B b where COL3 = a.COL4) 
   WHERE ROW_NUMBER = 2
  )

ORA-00904:“A”。“COL4”:无效 ID。

所以,a.COL4 在子子查询中是未知的,但我不知道如何解决这个问题。

/编辑。我想做什么?

对于 TABLE_A 中的每条记录,TABLE_B 中都有多条记录。然而,来自客户的新要求:TABLE_A 将获得 2 个新列,而 TABLE_B 将被删除。因此,子查询的第一条记录的表示将被写入第一个新字段,第二个字段也是如此。第一个记录很容易,因为 Mike C 的解决方案可以与 ROW_NUMBER = 1 一起使用。

示例行:

TABLE_A

| col0 | col1 | col2 | col3 | col4 |
------------------------------------
|      |      |dummy2|dummy3|   1  |
------------------------------------
|      |      |dummy4|dummy5|   2  |
------------------------------------

TABLE_B

| col1 | col2 | col3 |
----------------------
|  d   |name1 |   1  |
----------------------
|  d   |name2 |   1  |
----------------------
|  d   |name3 |   1  |
----------------------
|  d   |name4 |   2  |
----------------------


TABLE_A after update

| col0 | col1 | col2 | col3 | col4 |
------------------------------------
| name1| name2|dummy2|dummy3|   1  |
------------------------------------
| name4|      |dummy4|dummy5|   2  |
------------------------------------

【问题讨论】:

  • “where row_number=2”的目的是什么?你到底想做什么?
  • 正在编辑中。需要将 TABLE_B 中前两条记录的表示形式与 TABLE_A 中的字段匹配到 TABLE_A 中的两个新字段中。
  • 好的,所以是一个数据透视表,将 tableB 中的 2 行转换为 tableA 中的 2 列?可以考虑通过 CTAS 将 2 个表连接到一个数据透视表中来重新创建 tableA。什么 Oracle 版本?
  • 另外,一个简单的表结构示例和一些具有所需输出的测试行会有所帮助

标签: sql oracle sql-update ora-00904


【解决方案1】:
UPDATE TABLE_A a SET COL1 = 
  (SELECT b.COL2 FROM
    (SELECT ROWNUM AS ROW_NUMBER, b.COL2 FROM TABLE_B b, TABLE_A innerA WHERE COL3 = innerA.COL4) 
   WHERE ROW_NUMBER = 2
  )

【讨论】:

  • 内部选择将返回比我在原始查询中预期的更多的记录。它不会从 TABLE_B 返回与正在从 TABLE_A 更新的记录匹配的记录,而是从 TABLE_B 返回与来自 TABLE_A 的任何记录匹配的记录。
  • 嗯,好的。如果您在原始问题中描述您想要通过查询实现的目标,这可能会有所帮助。
【解决方案2】:

你能像这样消除其中一个子查询吗?

update TABLE_A a set COL1 = 
(SELECT b.COL2 FROM TABLE_B b where COL3 = a.COL4 AND ROWNUM = 2)

【讨论】:

    【解决方案3】:

    试试

    update TABLE_A a set COL1 = 
      (SELECT b.COL2 FROM
        (SELECT ROWNUM AS ROW_NUMBER, b.COL2 from TABLE_B b, TABLE_A a2 where b.COL3 = a2.COL4) 
       WHERE ROW_NUMBER = 2
      )
    

    我假设 COL3 来自表 b,为什么还要在子查询中选择 ROWNUM? WHERE 子句中只能是 2 个。

    【讨论】:

    • 啊,是的,看看rownum的问题。你到底想做什么?
    【解决方案4】:

    我认为这可能是解决您的问题的一个可能的解决方案,但根据您正在处理的数据量,它可能会非常慢,因为内部语句没有限制因素。

    update
      table_a upd
    set upd.col1 = (
      select
        sub.col2
      from
        (
          select
            rownum as row_number,
            b.col2 as col2,
            b.col3 as col3
          from
            table_a a,
            table_b b
          where b.col3 = a.col4
        ) sub
      where sub.row_number = 2
        and sub.col3       = upd.col4
    )
    

    【讨论】:

      【解决方案5】:

      我使用临时表解决了这个问题,在表 A 填满时从其中删除数据。

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 1970-01-01
        • 2016-05-13
        • 1970-01-01
        • 2012-07-20
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        相关资源
        最近更新 更多