Hibernate支持樂觀鎖。當(dāng)多個事務(wù)同時對數(shù)據(jù)庫表中的同一條數(shù)據(jù)操作時,如果沒有加鎖機制的話,就會產(chǎn)生臟數(shù)據(jù)(duty data)。Hibernate有2種機制可以解決這個問題:樂觀鎖和悲觀鎖。這里我們只討論樂觀鎖。
Hibernate樂觀鎖,能自動檢測多個事務(wù)對同一條數(shù)據(jù)進(jìn)行的操作,并根據(jù)先勝原則,提交第一個事務(wù),其他的事務(wù)提交時則拋出org.hibernate.StaleObjectStateException異常。
Hibernate樂觀鎖是怎么做到的呢?
我們先從Hibernate樂觀鎖的實現(xiàn)說起。要實現(xiàn)Hibenate樂觀鎖,我們首先要在數(shù)據(jù)庫表里增加一個版本控制字段,字段名隨意,比如就叫version,對應(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中對該數(shù)據(jù)修改
tx1.commit();
session1.close();
//事務(wù)1提交
Transaction tx2 = session2.beginTransaction();
//事務(wù)2開始
et2.setName(“Entity2”);
//事務(wù)2中對該數(shù)據(jù)修改
tx2.commit();
session2.close();
//事務(wù)2提交
在事務(wù)2提交時,因為它提交的數(shù)據(jù)比事務(wù)1提交后的數(shù)據(jù)舊,所以hibernate會拋出一個org.hibernate.StaleObjectStateException異常。
回到前面的問題,Hibernate怎么知道事務(wù)2提交的數(shù)據(jù)比事務(wù)1提交后的數(shù)據(jù)舊呢?
因為MyEntity有個version版本控制字段。
回頭看看上面的源代碼中的:
MyEntity et1 = session1.load(MyEntity.class, id);
MyEntity et2 = session2.load(MyEntity.class, id);
這里,et1.version==et2.version,比如此時version=1,
當(dāng)事務(wù)1提交后,該數(shù)據(jù)的版本控制字段version=version+1=2,而事務(wù)2提交時version=1<2所以Hibernate認(rèn)為事務(wù)2提交的數(shù)據(jù)為過時數(shù)據(jù),拋出異常。
這就是Hibernate樂觀鎖的原理機制。
我們已經(jīng)知道了Hibernate樂觀鎖是根據(jù)version的值來判斷數(shù)據(jù)是否過時,也就是說,在向數(shù)據(jù)庫update某數(shù)據(jù)時,必須保證該entity 里的version字段被正確地設(shè)置為update之前的值,否則hibernate樂觀鎖機制將無法根據(jù)version作出正確的判斷。
在我們的WEB應(yīng)用中,尤其應(yīng)該注意這個問題。