濮阳杆衣贸易有限公司

主頁 > 知識(shí)庫 > jsp hibernate 數(shù)據(jù)保存操作的原理

jsp hibernate 數(shù)據(jù)保存操作的原理

熱門標(biāo)簽:商丘百應(yīng)電話機(jī)器人有沒有效果 地圖標(biāo)注人員兼職 騰訊地圖標(biāo)注商戶改名注冊(cè)入駐 漯河辦理400電話 電話機(jī)器人的特色和創(chuàng)新 開封便宜外呼系統(tǒng)報(bào)價(jià) 淮南騰訊地圖標(biāo)注 黃石智能營銷電銷機(jī)器人效果 怎樣把地圖標(biāo)注出來
數(shù)據(jù)的保存,更新和刪除:
1、Session.save()方法:
Session.save()方法用于實(shí)體對(duì)象的持久化保存,也就是說當(dāng)執(zhí)行session.save()方法時(shí)會(huì)生成對(duì)應(yīng)的insert SQL語句,完成數(shù)據(jù)的保存。如下面的代碼:
User user=new User();
user.setName(“zx”);
Transaction tx=session.beginTransaction();
session.save(user);
tx.commit();
當(dāng)執(zhí)行到session.save()方法時(shí),Hibernate并不會(huì)馬上生成insert SQL語句來進(jìn)行數(shù)據(jù)的保存,而是當(dāng)稍后清理session的緩存時(shí)才有可能執(zhí)行insert SQL語句,那么session.save()方法到底會(huì)執(zhí)行哪些步驟呢?請(qǐng)看進(jìn)行了如下總結(jié):
一、 在session的內(nèi)部緩存中尋找保存對(duì)象,如果找到了,則認(rèn)為此數(shù)據(jù)已經(jīng)保存(曾經(jīng)執(zhí)行過insert操作),實(shí)體對(duì)象已經(jīng)處于persistent狀態(tài),直接返回。此時(shí)即使數(shù)據(jù)相比之前的狀態(tài)發(fā)生了變化,也將在事務(wù)提交時(shí)由臟數(shù)據(jù)檢查來判定是否需要執(zhí)行update操作。
二、 如果實(shí)體對(duì)象實(shí)現(xiàn)了lifecycle接口,那么將執(zhí)行待保存對(duì)象的onSave()方法。
三、 如果實(shí)體對(duì)象實(shí)現(xiàn)了Validatable接口,那么將會(huì)執(zhí)行相應(yīng)的validate()方法。
四、 如果存在攔截器對(duì)象,那么將會(huì)執(zhí)行Interceptor.onSave()方法。
五、 構(gòu)造insert SQL語句完成數(shù)據(jù)保存。
六、 數(shù)據(jù)保存成功后,設(shè)定實(shí)體對(duì)象的id為插入記錄的id。
七、 將保存后的實(shí)體對(duì)象納入Hibernate的內(nèi)部緩存(一級(jí)緩存)。注意Hibernate不會(huì)把保存后的實(shí)體對(duì)象納入二級(jí)緩存,因?yàn)閯倓偙4孢^的實(shí)體對(duì)象很可能在之后被修改,緩存的頻繁更新以及帶來的同步問題代價(jià),超出了緩存該對(duì)象所帶來的收益。
八、 最后如果該對(duì)象有關(guān)聯(lián)對(duì)象,那么將會(huì)遞歸處理該級(jí)聯(lián)對(duì)象。

1、 Session.update()方法:
前面我在實(shí)體對(duì)象狀態(tài)轉(zhuǎn)化部分曾經(jīng)講過,session.update()方法能夠?qū)⒁粋€(gè)處于游離狀態(tài)的對(duì)象,重新納入Hibernate的內(nèi)部緩存,變成持久化對(duì)象。如下面的代碼:
Configuration cfg = new Configuration();
SessionFactory sf=cfg. configure().buildSessionFactory();
Customer customer=new Customer(“zx”,27,images);//customer對(duì)象處于自由狀態(tài)
Session session=sf.openSession();

Transaction tx=session.beginTransaction();
session.save(customer);//保存后customer對(duì)象處于持久化狀態(tài)
session.flush();//清空緩存后customer對(duì)象處于游離狀態(tài)
tx.commit();
session.close();

Session session2=sf.openSession();
Transaction tx2=session2.beginTransaction();
session2.update(customer);//通過調(diào)用update()方法將游離狀態(tài)的customer對(duì)象,再次轉(zhuǎn)化成持久化狀態(tài)
session2.delete(customer);//調(diào)用delete()方法后,當(dāng)清空緩存時(shí),會(huì)將customer對(duì)象移出緩存,同時(shí)會(huì)在數(shù)據(jù)庫中生成delete事務(wù),來刪除customer對(duì)象對(duì)應(yīng)的數(shù)據(jù)記錄
tx.commit();
session.close();
那么這個(gè)方法到底執(zhí)行了哪些步驟呢?它會(huì)按照下面的步驟進(jìn)行操作:
一、 首先會(huì)在緩存中尋找需要更新的實(shí)體對(duì)象,如果找到就立刻返回,從這里我們可以看出如果對(duì)一個(gè)已經(jīng)處于persistent的實(shí)體對(duì)象執(zhí)行update()方法,將不會(huì)產(chǎn)生任何作用。
二、 然后當(dāng)提交事務(wù)進(jìn)行緩存清理時(shí),將會(huì)通過臟數(shù)據(jù)檢查,確定變化的屬性,然后生成update SQL語句完成數(shù)據(jù)的更新。
這里有一個(gè)問題我們要強(qiáng)調(diào)一下,那就是只要通過update()方法將一個(gè)游離對(duì)象與session相關(guān)聯(lián),那么不論這個(gè)游離的實(shí)體對(duì)象的屬性是否發(fā)生改變,都會(huì)執(zhí)行update SQL語句。如下面的代碼:

Transaction tx=session.beginTransaction();
session.update(customer);
tx.commit();
session.close();
在這段代碼中并沒有修改customer對(duì)象的任何屬性值,但是也會(huì)執(zhí)行一個(gè)update SQL語句,如果你希望在沒有改變實(shí)體對(duì)象屬性值的情況下不去執(zhí)行update SQL語句,那么你要開啟實(shí)體對(duì)象class>元素的”select-before-update”屬性,將其設(shè)置為”true”,這個(gè)屬性默認(rèn)為”false”。如下進(jìn)行配置:
class name=”com.neusoft.entity.Customer” table=”customer” select-before-update=”true”>
如果啟用了這個(gè)屬性配置,那么在清理session緩存之前,會(huì)首先執(zhí)行類似如下的一條SQL語句:
Select * from customer where id='1';
查詢處所有的customer實(shí)體在數(shù)據(jù)庫中對(duì)應(yīng)的屬性值,然后逐條與緩存中屬性值進(jìn)行比較,如果發(fā)生了改變,那么將會(huì)生成update操作進(jìn)行數(shù)據(jù)更新,如果沒有發(fā)生改變那么將不會(huì)進(jìn)行update操作。要跟據(jù)實(shí)際需求情況來決定是否開啟這個(gè)選項(xiàng),如果實(shí)體對(duì)象的屬性不會(huì)經(jīng)常發(fā)生改變,那么就應(yīng)該開啟這個(gè)選項(xiàng),以免執(zhí)行多余的update操作。如果實(shí)體對(duì)象的屬性會(huì)經(jīng)常發(fā)生改變,那么就沒必要開啟這個(gè)選項(xiàng),以免在執(zhí)行update操作前再執(zhí)行多余的select語句。

注:(1)、當(dāng)執(zhí)行對(duì)一個(gè)游離實(shí)體對(duì)象執(zhí)行session.update()操作時(shí),如果在數(shù)據(jù)庫中不存在這個(gè)實(shí)體對(duì)應(yīng)的紀(jì)錄,那么這個(gè)操作將會(huì)拋出異常。
(2)、當(dāng)執(zhí)行session.update()方法將一個(gè)游離對(duì)象與session關(guān)聯(lián)時(shí),如果此時(shí)在緩存中已經(jīng)存在了與該實(shí)體對(duì)象具有相同OID的持久化對(duì)象,那么這個(gè)方法會(huì)拋出異常。如下面代碼:
Customer customer1=new Customer(“1”,“zx”,27,images);
Session session1=sf.openSession();
Transaction tx=session1.beginTransaction();
session.save(customer1);
session.flush();
tx.commit();
session1.close();

Session session2=sf.openSession();
Transaction tx2=session2.beginTransaction();
Customer othercustomer=(Customer)session2.load(Customer.class,”1”);
session2.update(customer1)
tx2.commit();
session2.close();
當(dāng)再次將游離對(duì)象customer1與session2關(guān)聯(lián)時(shí),此時(shí)因?yàn)閘oad()操作,在緩存已經(jīng)加載了一個(gè)和customer1具有相同OID的othercustomer對(duì)象,此時(shí)由于Hibernate緩存的對(duì)象緩存機(jī)制不允許把OID相同的對(duì)象緩存,所以會(huì)拋出異常。
2、 Session.saveOrUpdate():
這個(gè)方法包含了save()方法和update()方法的特點(diǎn),如果傳入該方法的是一個(gè)游離對(duì)象,那么這個(gè)方法就會(huì)執(zhí)行update操作,如果傳入該方法的是一個(gè)臨時(shí)對(duì)象,那么這個(gè)方法就會(huì)執(zhí)行insert操作。這個(gè)方法幕后的工作原理如下:
a) 首先在緩存尋找,如果找到待保存的操作就直接返回。
b) 如果實(shí)體實(shí)現(xiàn)了攔截方法,那么就執(zhí)行isUnsaved()方法,判斷實(shí)體對(duì)象狀態(tài)。
c) 如果實(shí)體處于臨時(shí)狀態(tài)就執(zhí)行save(),如果實(shí)體處于游離狀態(tài)那么就執(zhí)行update()。
這里存在一個(gè)問題,那就是Hibernate是怎樣判斷一個(gè)實(shí)體是處于游離態(tài)還是臨時(shí)狀態(tài)的?如果實(shí)體滿足下面的一個(gè)條件,就認(rèn)為這個(gè)實(shí)體處于臨時(shí)狀態(tài)。
.Java對(duì)象的OID值為null。
.如果Java對(duì)象具有version屬性(將在并發(fā)加鎖部分講解)且為null。
.如果實(shí)體的id>設(shè)置了屬性u(píng)nsaved-value,而且OID值與unsaved-value值相等。
.如果實(shí)體的version屬性設(shè)置了unsaved-value,并且version屬性的值與unsaved-value值相等。
.如果實(shí)體實(shí)現(xiàn)了Interceptor,而且Interceptor.isUnsaved()方法返回true。
滿足這些條件中的一個(gè),這個(gè)實(shí)體就被認(rèn)為是臨時(shí)對(duì)象。
3、 Session.delete():
delete()方法用于從數(shù)據(jù)庫中刪除一個(gè)或一批實(shí)體所對(duì)應(yīng)的數(shù)據(jù),如果傳入的對(duì)象是持久化對(duì)象,么當(dāng)清理緩存時(shí),就會(huì)執(zhí)行delete操作。如果傳入的是游離對(duì)象,那么首先會(huì)使該對(duì)象與session相關(guān)聯(lián),然后當(dāng)清理緩存時(shí),再執(zhí)行delete操作。看如下代碼:
Session session=sessionFactory().openSession();
Transaction tx=session.beginTransaction();
Customer customer=(Customer)session.load(Customer.class,”1”);
session.delete(customer);//計(jì)劃執(zhí)行一條delete語句
tx.commit();//清理緩存,執(zhí)行一條delete語句
session.close();//關(guān)閉session,這時(shí)將會(huì)把customer對(duì)象從緩存中刪除。
如果上面的代碼中的customer對(duì)象是一個(gè)游離對(duì)象,那么當(dāng)執(zhí)行session.delete()方法時(shí),會(huì)首先將游離的customer對(duì)象與session相關(guān)聯(lián),然后再清理緩存時(shí),再執(zhí)行delete操作。如果你想一次刪除多條數(shù)據(jù),那么可以采用一個(gè)重載的delete()方法:delete(“from Customer c where c.id>'8' ”);這個(gè)方法可以刪除符合條件的所有數(shù)據(jù)。
您可能感興趣的文章:
  • JSP 開發(fā)之hibernate的hql查詢多對(duì)多查詢
  • JSP 開發(fā)之hibernate配置二級(jí)緩存的方法
  • JSP開發(fā)中hibernate框架的常用檢索方式總結(jié)
  • JSP 中Hibernate實(shí)現(xiàn)映射枚舉類型
  • jsp Hibernate批量更新和批量刪除處理代碼
  • jsp Hibernate入門教程
  • jsp Hibernate 函數(shù)簡介
  • jsp hibernate的分頁代碼
  • JSP開發(fā)之hibernate之單向多對(duì)一關(guān)聯(lián)的實(shí)例

標(biāo)簽:鄭州 亳州 紅河 馬鞍山 岳陽 大興安嶺 武威 拉薩

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《jsp hibernate 數(shù)據(jù)保存操作的原理》,本文關(guān)鍵詞  jsp,hibernate,數(shù)據(jù),保存,操作,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《jsp hibernate 數(shù)據(jù)保存操作的原理》相關(guān)的同類信息!
  • 本頁收集關(guān)于jsp hibernate 數(shù)據(jù)保存操作的原理的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章
    株洲市| 启东市| 扶沟县| 宜昌市| 淮阳县| 东兴市| 微博| 昔阳县| 互助| 涿州市| 嘉黎县| 万盛区| 正镶白旗| 临夏县| 建昌县| 冀州市| 太和县| 新泰市| 门头沟区| 瑞昌市| 南涧| 罗源县| 德清县| 中西区| 临武县| 大埔区| 定襄县| 泰顺县| 浮梁县| 巴彦县| 渝中区| 策勒县| 安阳市| 汽车| 浦县| 通河县| 遂溪县| 涟水县| 积石山| 乌拉特前旗| 易门县|