sql多用戶
sql多用戶訪問資料庫其實就是事務並發,會引起如下問題:x0dx0a1、臟讀:一個事務讀取到了另外一個事務沒有提交的數據x0dx0a事務1:更新一條數據x0dx0a事務2:讀取事務1更新的記錄x0dx0a事務1:調用commit進行提交x0dx0a此時事務2讀取到的數據是保存在資料庫內存中的數據,稱為臟讀。x0dx0a讀到的數據為臟數據x0dx0a詳細解釋:x0dx0a臟讀就是指:當一個事務正在訪問數據,並且對數據進行了修改,而這種修改還沒有提交到資料庫中,這時,x0dx0a另外一個事務也訪問這個數據,然後使用了這個數據。因為這個數據是還沒有提交的數據,那麼另外一個x0dx0a事務讀到的這個數據是臟數據,依據臟數據所做的操作可能是不正確的。x0dx0a2、不可重復讀:在同一事務中,兩次讀取同一數據,得到內容不同x0dx0a事務1:查詢一條記錄x0dx0a事務2:更新事務1查詢的記錄x0dx0a事務2:調用commit進行提交x0dx0a事務1:再次查詢上次的記錄x0dx0a此時事務1對同一數據查詢了兩次,可得到的內容不同,稱為不可重復讀。x0dx0a3、幻讀:同一事務中,用同樣的操作讀取兩次,得到的記錄數不相同x0dx0a事務1:查詢表中所有記錄x0dx0a事務2:插入一條記錄x0dx0a事務2:調用commit進行提交x0dx0a事務1:再次查詢表中所有記錄x0dx0a此時事務1兩次查詢到的記錄是不一樣的,稱為幻讀x0dx0a詳細解釋:x0dx0a幻讀是指當事務不是獨立執行時發生的一種現象,例如第一個事務對一個表中的數據進行了修改,x0dx0a這種修改涉及到表中的全部數據行。同時,第二個事務也修改這個表中的數據,這種修改是向表x0dx0a中插入一行新數據。那麼,以後就會發生操作第一個事務的用戶發現表中還有沒有修改的數據行,x0dx0a就好象發生了幻覺一樣。x0dx0a處理以上隔離級別的問題,採用如下方是:x0dx0a事務隔離五種級別:x0dx0aTRANSACTION_NONE不使用事務。x0dx0aTRANSACTION_READ_UNCOMMITTED允許臟讀。x0dx0aTRANSACTION_READ_COMMITTED防止臟讀,最常用的隔離級別,並且是大多數資料庫的默認隔離級別x0dx0aTRANSACTION_REPEATABLE_READ可以防止臟讀和不可重復讀,x0dx0aTRANSACTION_SERIALIZABLE可以防止臟讀,不可重復讀取和幻讀,(事務串列化)會降低資料庫的效率x0dx0a以上的五個事務隔離級別都是在Connection介面中定義的靜態常量,x0dx0a使用setTransactionIsolation(intlevel)方法可以設置事務隔離級別。x0dx0a如:con.setTransactionIsolation(Connection.REPEATABLE_READ);x0dx0a注意:事務的隔離級別受到資料庫的限制,不同的資料庫支持的的隔離級別不一定相同x0dx0a1臟讀:修改時加排他鎖,直到事務提交後才釋放,讀取時加共享鎖,讀取完釋放事務1讀取數據時加上共享鎖後(這樣在事務1讀取數據的過程中,其他事務就不會修改該數據),不允許任何事物操作該數據,只能讀取,之後1如果有更新操作,那麼會轉換為排他鎖,其他事務更無權參與進來讀寫,這樣就防止了臟讀問題。x0dx0a但是當事務1讀取數據過程中,有可能其他事務也讀取了該數據,讀取完畢後共享鎖釋放,此時事務1修改數據,修改完畢提交事務,其他事務再次讀取數據時候發現數據不一致,就會出現不可重復讀問題,所以這樣不能夠避免不可重復讀問題。x0dx0a2不可重復讀:讀取數據時加共享鎖,寫數據時加排他鎖,都是事務提交才釋放鎖。讀取時候不允許其他事物修改該數據,不管數據在事務過程中讀取多少次,數據都是一致的,避免了不可重復讀問題x0dx0a3幻讀問題:採用的是范圍鎖RangeSRangeS_S模式,鎖定檢索范圍為只讀,這樣就避免了幻影讀問題。