咱們在數據庫相關的開發中常常遇到這樣一個問題:向數據庫表中插入某條記錄,如果存在就對其進行更新。例如,有以下數據表user(id爲主鍵):sql
id | name | passwd |
1 | usr1 | pswd1 |
... | ... | ... |
向其插入id=1,name=usr1,passwd=pswd2的記錄,若存在id=1的記錄,則對其進行更新操做。
數據庫
這個問題在單鏈接訪問的條件下下很簡單,能夠先對記錄執行UPDATE操做,若影響的條數爲0,說明沒有此記錄,而後能夠放心大膽的進行INSERT操做。
併發
可是在多併發訪問的條件下,上述作法則存在同步問題。如果引入事務則又顯得有些小題大作,如果能有一條語句以原子操做的方式完成上述功能那即是極好的。現將MySQL與Oracle的處理辦法總結以下。
ide
一、MySQLspa
MySQL有兩種處理方式:REPLACE 和 ON DUPLICATE KEY
事務
(1)REPLACE
開發
REPLACE與INSERT的語法相仿,形式以下:
同步
replace into table(col1,col2,...) values(val1,val2,...);
例如:it
replace into user(id,name,passwd) values(1,'usr1','pawd2');
若是插入的記錄與表中原有的記錄不重複,則執行INSERT操做,影響的記錄數爲1;若是插入的記錄與表中原有的記錄重複,則先DELETE原有記錄,再執行INSERT,影響的記錄數爲2。
table
(2)ON DUPLICATE KEY
ON DUPLICATE KEY語句則是把要執行的INSERT語句和UPDATE語句鏈接在一塊兒。其形式以下:
insert_statement on duplicate key update_statement
例如:
insert into user(id,name,passwd) values(1,'usr1','pswd2') on duplicate key update name='usr1',passwd='pswd2';
若是插入的記錄與表中原有的記錄不重複,則執行前半部分的INSERT操做,影響的記錄數爲1;若是插入的記錄與表中原有的記錄重複,則執行後半部分的UPDATE操做,影響的記錄數爲2。
二、Oracle
Oracle則主要使用merge語句進行處理。例如:
merge into user using(select 1 id,'usr1','pswd2' from dual) t on (t.id=user.id) when matched then update set user.name='usr1',user.passwd='pswd2' when not matched then insert (user.id,user.name,user.passwd) values(1,'usr1','pswd2');