我們在數據庫相關的開發中經常遇到這樣一個問題:向數據庫表中插入某條記錄,若是存在就對其進行更新。例如,有如下數據表user(id爲主鍵):
id | name | passwd |
1 | usr1 | pswd1 |
... | ... | ... |
向其插入id=1,name=usr1,passwd=pswd2的記錄,若存在id=1的記錄,則對其進行更新操作。
這個問題在單連接訪問的條件下下很簡單,可以先對記錄執行UPDATE操作,若影響的條數爲0,說明沒有此記錄,然後可以放心大膽的進行INSERT操作。
但是在多併發訪問的條件下,上述做法則存在同步問題。若是引入事務則又顯得有些小題大做,若是能有一條語句以原子操作的方式完成上述功能那便是極好的。現將MySQL與Oracle的處理辦法總結如下。
1、MySQL
MySQL有兩種處理方式:REPLACE 和 ON DUPLICATE KEY
(1)REPLACE
REPLACE與INSERT的語法相仿,形式如下:
replace into table(col1,col2,...) values(val1,val2,...);
例如:
replace into user(id,name,passwd) values(1,'usr1','pawd2');
如果插入的記錄與表中原有的記錄不重複,則執行INSERT操作,影響的記錄數爲1;如果插入的記錄與表中原有的記錄重複,則先DELETE原有記錄,再執行INSERT,影響的記錄數爲2。
(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。
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');