Oracle數(shù)據(jù)庫(kù)兩表關(guān)聯(lián)更新的問(wèn)題
先放結(jié)論
UPDATE 需要更新的表名 B SET B.更新字段1 = '更新內(nèi)容1', B.更新字段2 = '更新內(nèi)容2' WHERE EXISTS (SELECT 1 FROM 關(guān)聯(lián)表名 A WHERE A.關(guān)聯(lián)字段 = B.關(guān)聯(lián)字段 AND A.篩選字段1 = '篩選字段1' AND A.篩選字段2 = '篩選字段2' AND B.篩選字段3 = '篩選字段3' );
問(wèn)題分析
需要寫(xiě)一個(gè)更新語(yǔ)句,但是更新的判斷條件是兩個(gè)表關(guān)聯(lián)查詢(xún)出來(lái)的
踩坑一
一開(kāi)始是打算UPDATE寫(xiě)兩個(gè)表,然后WHERE里面兩表關(guān)聯(lián),就像這樣:
--錯(cuò)誤sql UPDATE 表名1 A,表名2 B SET A.更新字段1 = '更新內(nèi)容1', A.更新字段2 = '更新內(nèi)容2' WHERE A.關(guān)聯(lián)字段 = B.關(guān)聯(lián)字段 AND A.篩選字段1 = '篩選字段1' AND A.篩選字段2 = '篩選字段2' AND B.篩選字段3 = '篩選字段3'
但是一直報(bào)錯(cuò)
很明顯也不是缺失SET 關(guān)鍵字的問(wèn)題
踩坑二
之后搜索怎么在Oracel關(guān)聯(lián)兩表更新。發(fā)現(xiàn)了下面這種寫(xiě)法
--正常sql UPDATE 更新表名 A SET A.更新字段1 = '更新內(nèi)容1' WHERE EXISTS (SELECT 1 FROM 關(guān)聯(lián)表名 B WHERE A.關(guān)聯(lián)字段 = B.關(guān)聯(lián)字段)
但是也是沒(méi)太仔細(xì)看,加上思維固化,寫(xiě)成了下面這種情況
--錯(cuò)誤sql UPDATE 更新表名 A SET A.更新字段1 = '更新內(nèi)容1' WHERE EXISTS (SELECT 1 FROM 更新表名 A, 關(guān)聯(lián)表名 B WHERE A.關(guān)聯(lián)字段 = B.關(guān)聯(lián)字段 AND A.篩選字段1 = '篩選字段1')
可以看到在FROM里面比搜索出來(lái)的sql語(yǔ)句,多了更新的表名A
顯而易見(jiàn),還會(huì)有點(diǎn)問(wèn)題
但是這次并不是報(bào)錯(cuò),而是篩選不是自己預(yù)想的篩選,導(dǎo)致更新的數(shù)據(jù)非常多
像下圖一樣,更新語(yǔ)句跑了1分半,更新了46萬(wàn)條數(shù)據(jù)。。。
明顯是有問(wèn)題的(實(shí)際上只需要更新幾十條),被更新的B表實(shí)際上也只有26萬(wàn)條數(shù)據(jù),不知道46萬(wàn)條更新怎么來(lái)的
只能先回滾更新
通過(guò)對(duì)比發(fā)現(xiàn)是FROM多寫(xiě)了個(gè)更新的表名,修改之后可以正常的更新,沒(méi)有問(wèn)題
踩坑三
既然已經(jīng)可以?xún)杀黻P(guān)聯(lián)查詢(xún),更新其中一個(gè)表了,那能不能同時(shí)更新兩個(gè)表呢?
想當(dāng)然的認(rèn)為sql應(yīng)該是這樣的
UPDATE 更新表1 A SET A.更新字段1 = 'A表更新內(nèi)容1' ,B.更新字段2 = 'B表更新內(nèi)容2' WHERE EXISTS (SELECT 1 FROM 更新表2 B WHERE A.關(guān)聯(lián)字段 = B.關(guān)聯(lián)字段 AND A.篩選字段 = '篩選字段')
于是寫(xiě)出了下面的sql,但是事實(shí)證明這樣寫(xiě)是不行的
語(yǔ)句分析
UPDATE 需要更新的表名 B SET B.更新字段1 = '更新內(nèi)容1', B.更新字段2 = '更新內(nèi)容2' WHERE EXISTS (SELECT 1 FROM 關(guān)聯(lián)表名 A WHERE A.關(guān)聯(lián)字段 = B.關(guān)聯(lián)字段 AND A.篩選字段1 = '篩選字段1' AND A.篩選字段2 = '篩選字段2' AND B.篩選字段3 = '篩選字段3' );
A表和B表都有相同的關(guān)聯(lián)字段,并且通過(guò)篩選字段校驗(yàn)。那么就有SELECT 1,那么EXISTS就會(huì)返回TRUE;最后進(jìn)行更新操作。
EXISTS
EXISTS用于檢查子查詢(xún)是否至少會(huì)返回一行數(shù)據(jù),該子查詢(xún)實(shí)際上并不返回任何數(shù)據(jù),而是返回值True或False
EXISTS(包括 NOT EXISTS )子句的返回值是一個(gè)BOOL值。 EXISTS內(nèi)部有一個(gè)子查詢(xún)語(yǔ)句(SELECT … FROM…), 我將其稱(chēng)為EXISTS的內(nèi)查詢(xún)語(yǔ)句。其內(nèi)查詢(xún)語(yǔ)句返回一個(gè)結(jié)果集。 EXISTS子句根據(jù)其內(nèi)查詢(xún)語(yǔ)句的結(jié)果集空或者非空,返回一個(gè)布爾值。
此時(shí)在該更新sql中
--EXISTS的內(nèi)查詢(xún)語(yǔ)句為: SELECT 1 FROM 關(guān)聯(lián)表名 A WHERE A.關(guān)聯(lián)字段 = B.關(guān)聯(lián)字段 AND A.篩選字段1 = '篩選字段1' AND A.篩選字段2 = '篩選字段2' AND B.篩選字段3 = '篩選字段3' --WHERE判斷EXISTS返回的BOOL值來(lái)決定是否更新 WHERE EXISTS
SELECT 1
當(dāng)我們只關(guān)心數(shù)據(jù)表有多少記錄行而不需要知道具體的字段值時(shí)
,SELECT 1 FROM TABLE
是一個(gè)很不錯(cuò)的SQL語(yǔ)句寫(xiě)法,它通常用于子查詢(xún)。
SELECT 1 FROM TABLE可以減少系統(tǒng)開(kāi)銷(xiāo),提高運(yùn)行效率。因?yàn)榇藭r(shí)數(shù)據(jù)庫(kù)就不會(huì)去檢索數(shù)據(jù)表里每條具體的記錄和每條記錄里每個(gè)具體的字段值并將它們放到內(nèi)存里,而是查詢(xún)到有多少記錄行存在就輸出多少個(gè)“1”,每個(gè)“1”代表有1行記錄。
選用數(shù)字1是因?yàn)樗加玫膬?nèi)存空間最小,用數(shù)字0的效果也一樣,即:SELECT 0 FROM TABLE。
從效率上來(lái)說(shuō),SELECT 1 > SELECT 字段 > SELECT *
在子查詢(xún)中的應(yīng)用
常規(guī)寫(xiě)法
SELECT * FROM t1 a WHERE EXISTS (SELECT * FROM t2 b WHERE a.id = b.id)
更優(yōu)寫(xiě)法
SELECT * FROM t1 a WHERE EXISTS (SELECT 1 FROM t2 b WHERE a.id = b.id)
到此這篇關(guān)于Oracle數(shù)據(jù)庫(kù)兩表關(guān)聯(lián)更新的文章就介紹到這了,更多相關(guān)Oracle兩表關(guān)聯(lián)更新內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Oracle中的ALL_TAB_COLUMNS視圖語(yǔ)句詳解
ALL_TAB_COLUMNS 是 Oracle 數(shù)據(jù)庫(kù)的一個(gè)數(shù)據(jù)字典視圖,用于提供關(guān)于數(shù)據(jù)庫(kù)中所有可見(jiàn)表的列信息,這篇文章主要介紹了Oracle中的ALL_TAB_COLUMNS視圖語(yǔ)句,需要的朋友可以參考下2024-08-08Oracle 12c修改字符集的方法(解決數(shù)據(jù)導(dǎo)入后中文亂碼及ORA-12899錯(cuò)誤)
之前在Windows上安裝的Oracle,現(xiàn)在遷移到Linux上,把dmp文件導(dǎo)入Linux的時(shí)候發(fā)現(xiàn)字段的注釋和存儲(chǔ)過(guò)程中的中文是問(wèn)號(hào)?,而且導(dǎo)入的時(shí)候還會(huì)報(bào)ORA-12899錯(cuò)誤,其實(shí)這些都是字符集問(wèn)題,所以本文給大家介紹了Oracle 12c修改字符集的方法,需要的朋友可以參考下2024-04-04Oracle針對(duì)數(shù)據(jù)庫(kù)某一行進(jìn)行操作的時(shí)候,如何將這一行加行鎖
Oracle針對(duì)數(shù)據(jù)庫(kù)某一行進(jìn)行操作的時(shí)候,如何將這一行加行鎖的實(shí)現(xiàn)方法2009-02-02Oracle RAC環(huán)境下的阻塞(blocking blocked)介紹和實(shí)例演示
這篇文章主要介紹了Oracle RAC環(huán)境下的阻塞(blocking blocked)介紹和實(shí)例演示本文提供了2個(gè)查詢(xún)腳本,并給出實(shí)例演示那些session為阻塞者,哪些為被阻塞者,需要的朋友可以參考下2014-09-09Oracle數(shù)據(jù)庫(kù)中如何按天、周、月、季、年統(tǒng)計(jì)數(shù)據(jù)
我們經(jīng)常遇到一些需求,需要我們?cè)趕ql語(yǔ)句中對(duì)日期進(jìn)行分類(lèi)統(tǒng)計(jì),下面這篇文章主要給大家介紹了關(guān)于Oracle數(shù)據(jù)庫(kù)中如何按天、周、月、季、年統(tǒng)計(jì)數(shù)據(jù)的相關(guān)資料,需要的朋友可以參考下2024-03-03Oracle數(shù)據(jù)庫(kù)執(zhí)行腳本常用命令小結(jié)
這篇文章主要介紹了Oracle數(shù)據(jù)庫(kù)執(zhí)行腳本常用命令小結(jié)的相關(guān)資料,需要的朋友可以參考下2015-12-12連接Oracle數(shù)據(jù)庫(kù)失敗(ORA-12514)故障排除全過(guò)程
Oracle連接失敗是指在使用Oracle數(shù)據(jù)庫(kù)進(jìn)行開(kāi)發(fā)的過(guò)程中,服務(wù)器端無(wú)法與客戶(hù)端連接,從而導(dǎo)致Oracle連接無(wú)法成功,影響開(kāi)發(fā)的效率,下面這篇文章主要給大家介紹了關(guān)于連接Oracle數(shù)據(jù)庫(kù)失敗(ORA-12514)故障排除的相關(guān)資料,需要的朋友可以參考下2023-05-05Oracle出現(xiàn)超出打開(kāi)游標(biāo)最大數(shù)的解決方法
這篇文章主要介紹了Oracle出現(xiàn)超出打開(kāi)游標(biāo)最大數(shù)的解決方法,涉及針對(duì)Oracle游標(biāo)位置的判斷與處理技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2016-06-06AWR 深入分析( Automatic Workload Repository )
本篇文章,小編為大家介紹一下關(guān)于A(yíng)WR 深入分析( Automatic Workload Repository )有需要的朋友可以參考一下2013-04-04