http://blog.csdn.net/yuzhic/article/details/1896878工具
http://blog.csdn.net/macle2010/article/details/5980965spa
该命令使用一条语句从一个或者多个数据源中完成对表的更新和插入数据. ORACLE 9i 中,使用此命令必须同时指定UPDATE 和INSERT 关键词,ORACLE 10g 作了以下改动。.net
1,insert 和update是可选的 2,UPDATE 和INSERT 后面能够跟WHERE 子句 3,在ON条件中可使用常量来insert 全部的行到目标表中,不须要链接到源表和目标表 4,UPDATE 子句后面能够跟delete 来去除一些不须要的行。blog
举例:ip
- createtable PRODUCTS
- (
- PRODUCT_ID INTEGER,
- PRODUCT_NAME VARCHAR2(60),
- CATEGORY VARCHAR2(60)
- );
-
- insertinto PRODUCTS values (1501, 'VIVITAR 35MM', 'ELECTRNCS');
- insertinto PRODUCTS values (1502, 'OLYMPUS IS50', 'ELECTRNCS');
- insertinto PRODUCTS values (1600, 'PLAY GYM', 'TOYS');
- insertinto PRODUCTS values (1601, 'LAMAZE', 'TOYS');
- insertinto PRODUCTS values (1666, 'HARRY POTTER', 'DVD');
- commit;
-
- createtable NEWPRODUCTS
- (
- PRODUCT_ID INTEGER,
- PRODUCT_NAME VARCHAR2(60),
- CATEGORY VARCHAR2(60)
- );
-
- insertinto NEWPRODUCTS values (1502, 'OLYMPUS CAMERA', 'ELECTRNCS');
- insertinto NEWPRODUCTS values (1601, 'LAMAZE', 'TOYS');
- insertinto NEWPRODUCTS values (1666, 'HARRY POTTER', 'TOYS');
- insertinto NEWPRODUCTS values (1700, 'WAIT INTERFACE', 'BOOKS');
- commit;
- 1,可省略的update 或者insert
- MERGE INTO products p
- 2 USING newproducts np
- 3 ON (p.product_id = np.product_id)
- 4 WHEN MATCHED THEN
- 5 UPDATE
- 6 SET p.product_name = np.product_name,
- 7 p.category = np.category;
使用表newproducts中的product_name 和category字段来更新表products 中相同product_id的product_name 和category.string
2,当条件不知足的时候把newproducts表中的数据INSERT 到表products中。it
- MERGE INTO products p
- USING newproducts np
- ON (p.product_id = np.product_id)
- WHENNOT MATCHED THEN
- INSERT
- VALUES (np.product_id, np.product_name,
- np.category);
3,带条件的insert 和updateio
- MERGE INTO products p
- USING newproducts np
- ON (p.product_id = np.product_id)
- WHEN MATCHED THEN
- UPDATE
- SET p.product_name = np.product_name
- WHERE p.category = np.category;
insert 和update 都带有where 字句table
- MERGE INTO products p
- USING newproducts np
- ON (p.product_id = np.product_id)
- WHEN MATCHED THEN
- UPDATE
- SET p.product_name = np.product_name,
- p.category = np.category
- WHERE p.category = 'DVD'
- WHENNOT MATCHED THEN
- INSERT
- VALUES (np.product_id, np.product_name, np.category)
- WHERE np.category != 'BOOKS'
4,无条件的insertclass
- MERGE INTO products p
- USING newproducts np
- ON (1=0)
- WHENNOT MATCHED THEN
- INSERT
- VALUES (np.product_id, np.product_name, np.category)
- WHERE np.category = 'BOOKS'
5,delete 子句
1 merge into products p 2 using newproducts np 3 on(p.product_id = np.product_id) 4 when matched then 5 update 6 set p.product_name = np.product_name 7 delete where category = 'macle1_cate';
select *
from products;
PRODUCT_ID PRODUCT_NAME CATEGORY --------------------------------------- -------------------- -------------------- 1502 macle22 macle2_cate 1503 macle3 macle2_cate 1504 macle macle1_cate 1505 macle5 macle5_cate
1504 中的macle1_cate 知足delete where,可是不知足 on 中的条件,因此没有被删除。!!!!!!重点
-----------------------------------------------
动机:
想在Oracle中用一条SQL语句直接进行Insert/Update的操做。
说明:
在进行SQL语句编写时,咱们常常会遇到大量的同时进行Insert/Update的语句 ,也就是说当存在记录时,就更新(Update),不存在数据时,就插入(Insert)。
实战:
接下来咱们有一个任务,有一个表T,有两个字段a,b,咱们想在表T中作Insert/Update,若是存在,则更新T中b的值,若是不存在,则插入一条记录。在Microsoft的SQL语法中,很简单的一句判断就能够了,SQL Server中的语法以下:
if exists(select 1 from T where T.a='1001' ) update T set T.b=2 Where T.a='1001' else insert into T(a,b) values('1001',2);
以上语句代表当T表中若是存在a='1001' 的记录的话,就把b的值设为2,不然就Insert一条a='100',b=2的记录到T中。
可是接下来在Oracle中就遇到麻烦了,记得在Oracle 9i以后就有一条Merge into 的语句能够同时进行Insert 和Update的吗,Merge的语法以下:
MERGE
INTO
table_name alias1
USING (
table
|
view
|
sub_query) alias2
ON
(
join
condition)
WHEN
MATCHED
THEN
UPDATE
table_name
SET
col1
=
col_val1,
col2
=
col2_val
WHEN
NOT
MATCHED
THEN
INSERT
(column_list)
VALUES
(column_values);
上面的语法你们应该都容易懂吧,那咱们按照以上的逻辑再写一次。
MERGE
INTO
T T1
USING (
SELECT
a,b
FROM
T
WHERE
t.a
=
'
1001
'
) T2
ON
( T1.a
=
T2.a)
WHEN
MATCHED
THEN
UPDATE
SET
T1.b
=
2
WHEN
NOT
MATCHED
THEN
INSERT
(a,b)
VALUES
(
'
1001
'
,
2
);
以上的语句貌似很对是吧
,实际上,该语句只能进行更新,而没法进行Insert,错误在哪里呢?
其实在Oracle中Merge语句原先是用来进行整表的更新用的,也就是ETL工具比较经常使用的语法,重点是在Using上。
用中文来解释Merge语法,就是:
在alias2中Select出来的数据,每一条都跟alias1进行 ON (join condition)的比较,若是匹配,就进行更新的操做(Update),若是不匹配,就进行插入操做(Insert)。
所以,严格意义上讲,”在一个同时存在Insert和Update语法的Merge语句中,总共Insert/Update的记录数,就是Using语句中alias2的记录数。”
以上这句话也就很好的解释了在上面写的语句为什么只能进行Update,而不能进行Insert了,由于都Select不到数据,如何能进行Insert呢:)
接下来要改为正确的语句就容易多了,以下:
MERGE
INTO
T T1
USING (
SELECT'1001'AS a,2AS b FROM
dual) T2
ON
( T1.a
=
T2.a)
WHEN
MATCHED
THEN
UPDATE
SET
T1.b
=
T2.b
WHEN
NOT
MATCHED
THEN
INSERT
(a,b)
VALUES
(T2.a,T2.b);
查询结果,OK!
注意:
若是不懂Merge语句的原理,Merge语句是一条比较危险的语句,特别是在您只想更新一条记录的时候,由于不经意间,你可能就把整表的数据都Update了一遍.....汗!!!
我曾经犯过的一个错误以下所示,你们看出来是什么问题了吗?
MERGE
INTO
T T1
USING (
SELECT
Count
(
*
) cnt
FROM
T
WHERE
T.a
=
'
1001
'
) T2
ON
(T2.cnt
>
0
)
WHEN
MATCHED
THEN
UPDATE
SET
T1.b
=
T2.b
WHEN
NOT
MATCHED
THEN
INSERT
(a,b)
VALUES
(T2.a,T2.b);