ORACLE批量更新方法

转载 2013年12月03日 08:15:52

第一种方法 就是update tab a set a.name=(select b.name from b where a.id=b.id) where exists (select 1 from b where a.id=b.id)   但是我不知道为啥要加后面 exists 

sql%rowcount的数据量(不管你是否更新到)就是select count(*) from table_name的数据量啦,即全表更新
第二种就是 merge into tab
  using ()  on (条件)
  when match then
   do update
  when no match then
  do insert       

第三种估计好多人不知道    update(select a.name,b.name from a,b where a.id=b.id) set a.name=b.name;  但是这种情况下要保证b.id要唯一 最好还能建立索引啥的

第二种oracle merge into的使用

分类: SQL Programing 439人阅读 评论(0)收藏 举报

在做报表的时候,遇到这么一个问题,由于数据量大,数据上传时间问题,经常要处理更新几天前的数据,这时需要涉及UPDATE或者INSERT两个操作,这时推荐用MERGE INTO,但用这个时候需注意一个问题,我先来一个测试:


SQL> select * from ytrep.tab1;
 
COL_A      COL_B      COL_C
---------- ---------- ----------
1          A          
2          B          
3          C          
4          D          
5          E          
6          F          
7          G          
8          H    

      
 
8 rows selected


SQL> select * from ytrep.tab2;
 
COL_A      COL_B      COL_C      COL_D
---------- ---------- ---------- ----------
1          A          A1         1
1          A          A1         2

2          B          B1         1
3          C          C1         1
4          D          D1         1
5          E          E1         2
5          E          E2         1
6          F          F1         1
6          F          F1         2
9          G          G1         1
10         H          H1         1
 
11 rows selected

有以上两表,其中需要根据ytrep.tab2表的col_a及col_b值来更新ytrep.tab1的col_c值,这时,我们来执行以下SQL:

Merge into ytrep.tab1 p
using (
        select t.col_A, t.col_B, col_C
        from ytrep.tab2 t
      ) tmp
on (p.col_A= tmp.col_A and p.col_B= tmp.col_B)
  when not matched then
    insert (col_A, col_B, col_C) values (tmp.col_a, tmp.col_B, tmp.col_C)
  when matched then
    update set p.col_c = tmp.col_c;

提示:

ORA-30926: 无法在源表中获得一组稳定的行

ORA-30926: unable to get a stable set of rows in the source tables


好了,现在我们来分析为什么会出现这种错误,在ON条件里,我们可以看到p.col_A= tmp.col_A and p.col_B= tmp.col_B,

当tmp表返回记录后,ytrep.tab1取第一条记录 col_a = '1' and col_b = 'A' 与tmp表关联时,发现tmp有两条匹配的记录,这时ORACLE就无法判断到底UPDATE SET时应该取TMP表的哪条记录的COL_C值是A1还是A2了,这时ORACLE就返回以上错误;

这时,我们就可以推断,TMP结果集里返回的在ON条件里有涉及字段的值必须是唯一的,否则肯定报以上ORA-30926错误;


这时,有人也许会问,那到底该怎么写呢?

可分为两种情况:

1、一种是通过将TMP表的数据GROUP BY,取得MIN或MAX值;

Merge into ytrep.tab1 p
using (
        select t.col_A, t.col_B, max(col_C) col_C  --min(col_C) col_C
        from ytrep.tab2 t
        group by t.col_a, t.col_b
      ) tmp
on (p.col_A= tmp.col_A and p.col_B= tmp.col_B)
  when not matched then
    insert (col_A, col_B, col_C) values (tmp.col_a, tmp.col_B, tmp.col_C)
  when matched then
    update set p.col_c = tmp.col_c;


SQL> select * from ytrep.tab1;
 
COL_A      COL_B      COL_C
---------- ---------- ----------
1          A          A1
2          B          B1
3          C          C1
4          D          D1
5          E          E2
6          F          F1
7          G          
8          H          
10         H          H1
9          G          G1

10 rows selected


2、一种是分组后再根据某列值排序取第一行或最后一行值;

Merge into ytrep.tab1 p
using (
        select distinct t.col_A, t.col_B, first_value(col_c) over (partition by t.col_a, t.col_b order by t.col_d asc) col_c
        from ytrep.tab2 t
      ) tmp
on (p.col_A= tmp.col_A and p.col_B= tmp.col_B)
  when not matched then
    insert (col_A, col_B, col_C) values (tmp.col_a, tmp.col_B, tmp.col_C)
  when matched then
    update set p.col_c = tmp.col_c;


 
SQL> select * from ytrep.tab1;
 
COL_A      COL_B      COL_C
---------- ---------- ----------
1          A          A1
2          B          B1
3          C          C1
4          D          D1
5          E          E2
6          F          F1
7          G          
8          H          
10         H          H1
9          G          G1

10 rows selected


以上两个SQL,大家可以通过改变order by排序方式对比一下结果。


以上只是本人学习测试结果,如有不同意见或或好的建议,可以留言。谢谢!

 

 

 

相关文章推荐

ORACLE批量更新四种方法比较

软件环境 Windows 2000 + ORACLE9i 硬件环境 CPU 1.8G + RAM 512M 现在我们有2张表 如下:T1--大表 10000笔 T1_FK_ID T2--小表 5000...

Oracle的update语句优化研究 批量更新

Oracle的update语句优化研究 一、 update语句的语法与原理 1. 语法 单表:UPDATE 表名称 SET 列名称 = 新值 WHERE 列名称 = 某值 如:update t_joi...

Oracle批量Update记录

工作中经常用到Oracle批量更新记录,做为老手也怕出错,总之要小心再小心,确保数据批量更新正确。 下面举一个例子: 1、创建两张结构类似的表,建表语句如下: create table jayt1( ...

Otter-入门篇1(阿里开源项目Otter介绍)

Otter-入门篇1(阿里开源项目Otter介绍)前言呜啦啦啦啦!今天笔者又来开坑了,这次开坑的对象呢是阿里的一个开源项目Otter,Otter它是一个数据同步解决方案,可以解决本地跨网络跨机房跨地域...

oracle 批量更新四种方法比较

转ORACLE批量更新四种方法比较软件环境 Windows 2000 + ORACLE9i 硬件环境 CPU 1.8G + RAM 512M现在我们有2张表 如下: T1–大表 10000笔 T...

MyEclipse5.5 Generating Artifacts错误

MyEclipse5.5生成映射文件出现Generating Artifacts错误使用 MyEclipse5.5+classes12.jar+Hibernate3建立了 Hibernate3 的 P...

Oracle学习笔记(四):多表查询(二)

为了说明外连接的作用,我们做一个表来说明:

【Oracle批量更新】根据一个大表批量更新另一大表的方法比较

【问题】现在有两个千万级别的结构相同数据不同数据表T_SMS_PHONENO(目的表),T_SMS_PHONENO2(源表),根据源表数据更新目的表的数据。【分析】根据经验,更新方法一般有以下几种:1...

Oracle中采用存储过程的方式批量更新数据

采用存储过程的方式批量更新数据,Oracle中也可采用merge-update的方式更新,采用批量提交更新方式会更快 DECLARE MAX_ROWS NUMBER DEF...

oracle批量update

需求: 将t2(t_statbuf)表中id和t1(T_Mt)表相同的记录更新进t1表。 1.错误的写法: update table_name t1 set (a,b,c)=( select ...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)