Hibernate支持樂觀鎖。當(dāng)多個(gè)事務(wù)同時(shí)對(duì)數(shù)據(jù)庫表中的同一條數(shù)據(jù)操作時(shí),如果沒有加鎖機(jī)制的話,就會(huì)產(chǎn)生臟數(shù)據(jù)(duty data)。Hibernate有2種機(jī)制可以解決這個(gè)問題:樂觀鎖和悲觀鎖。這里我們只討論樂觀鎖。
Hibernate樂觀鎖,能自動(dòng)檢測(cè)多個(gè)事務(wù)對(duì)同一條數(shù)據(jù)進(jìn)行的操作,并根據(jù)先勝原則,提交第一個(gè)事務(wù),其他的事務(wù)提交時(shí)則拋出org.hibernate.StaleObjectStateException異常。
Hibernate樂觀鎖是怎么做到的呢?
我們先從Hibernate樂觀鎖的實(shí)現(xiàn)說起。要實(shí)現(xiàn)Hibenate樂觀鎖,我們首先要在數(shù)據(jù)庫表里增加一個(gè)版本控制字段,字段名隨意,比如就叫version,對(duì)應(yīng)hibernate類型只能為 long,integer,short,timestamp,calendar,也就是只能為數(shù)字或timestamp類型。然后在hibernate mapping里作如下類似定義:
<version name="version"
column="VERSION"
type="integer"
/>
告訴Hibernate version作為版本控制用,交由它管理。
當(dāng)然在entity class里也需要給version加上定義,定義的方法跟其他字段完全一樣。
private Integer version;
…
// setVersion() && getVersion(Integer)
Hibernate樂觀鎖的的使用:
Session session1 = sessionFactory.openSession();
Session session2 = sessionFactory.openSession();
MyEntity et1 = session1.load(MyEntity.class, id);
MyEntity et2 = session2.load(MyEntity.class, id);
//這里 et1, et2為同一條數(shù)據(jù)
Transaction tx1 = session1.beginTransaction();
//事務(wù)1開始
et1.setName(“Entity1”);
//事務(wù)1中對(duì)該數(shù)據(jù)修改
tx1.commit();
session1.close();
//事務(wù)1提交
Transaction tx2 = session2.beginTransaction();
//事務(wù)2開始
et2.setName(“Entity2”);
//事務(wù)2中對(duì)該數(shù)據(jù)修改
tx2.commit();
session2.close();
//事務(wù)2提交
在事務(wù)2提交時(shí),因?yàn)樗峤坏臄?shù)據(jù)比事務(wù)1提交后的數(shù)據(jù)舊,所以hibernate會(huì)拋出一個(gè)org.hibernate.StaleObjectStateException異常。
回到前面的問題,Hibernate怎么知道事務(wù)2提交的數(shù)據(jù)比事務(wù)1提交后的數(shù)據(jù)舊呢?
因?yàn)?/span>MyEntity有個(gè)version版本控制字段。
回頭看看上面的源代碼中的:
MyEntity et1 = session1.load(MyEntity.class, id);
MyEntity et2 = session2.load(MyEntity.class, id);
這里,et1.version==et2.version,比如此時(shí)version=1,
當(dāng)事務(wù)1提交后,該數(shù)據(jù)的版本控制字段version=version+1=2,而事務(wù)2提交時(shí)version=1<2所以Hibernate認(rèn)為事務(wù)2提交的數(shù)據(jù)為過時(shí)數(shù)據(jù),拋出異常。
這就是Hibernate樂觀鎖的原理機(jī)制。
我們已經(jīng)知道了Hibernate樂觀鎖是根據(jù)version的值來判斷數(shù)據(jù)是否過時(shí),也就是說,在向數(shù)據(jù)庫update某數(shù)據(jù)時(shí),必須保證該entity 里的version字段被正確地設(shè)置為update之前的值,否則hibernate樂觀鎖機(jī)制將無法根據(jù)version作出正確的判斷。
在我們的WEB應(yīng)用中,尤其應(yīng)該注意這個(gè)問題。