您好,登錄后才能下訂單哦!
我們知道hibernate的核心就是對(duì)數(shù)據(jù)庫的操作,里面的核心接口就是org.hibernate.Session接口。要想對(duì)數(shù)據(jù)庫操作我們就要理清楚對(duì)象在整個(gè)操作中的所屬的狀態(tài)(Transient,Persistent,Detached)。就像馬士兵老師在視頻中所說的,我們并不必死摳這些字眼,我們通過自己編寫測試類就可以他們之間不同的區(qū)別。
其實(shí)三種狀態(tài)各自的不必總結(jié)那么多,只是一個(gè)重要的地方就是Transient狀態(tài)里面的對(duì)象是沒有id的。
session中常用的方法是save(),update(),saveOrUpdate(),get(),load(),delete(),clear(),flush()這些
其中大學(xué)期間我其實(shí)對(duì)get和load方法都不是太熟悉,死記硬背才在考試中答題,現(xiàn)在想起完全沒有這個(gè)必要。我們可以根據(jù)配置好的hibernate環(huán)境來進(jìn)行junit測試。
首先我們進(jìn)行g(shù)et()方法進(jìn)行測試。
這里我們首先建立一個(gè)實(shí)體類,StudentEntity.Java 和數(shù)據(jù)庫中student所對(duì)應(yīng)
package com.cwnu.entities; import javax.persistence.*; import java.io.Serializable; /** * Created by yangy on 2015/12/19. */ @Table(name = "student") @Entity public class StudentEntity implements Serializable{ private int id; private String name; @Id @Column(name = "id") @GeneratedValue(strategy = GenerationType.AUTO) private int getId() { return id; } public void setId(int id) { this.id = id; } @Column(name = "name") public String getName() { return name; } public void setName(String name) { this.name = name; } }
junit類都是idea編譯環(huán)境給我們配置好了,這里我們要注意,要使用getCurrentSession()來操作的話,我們首先要在Junit類中加上@Transaction注解,下面是相關(guān)的測試方法:
@Test public void testGetOrLoad() { Session session = sessionFactory.getCurrentSession(); StudentEntity student = (StudentEntity)session.get(StudentEntity.class, 1); //StudentEntity student_load = (StudentEntity)session.load(StudentEntity.class,1); String name = student.getName(); }
先測試get方法,我們發(fā)現(xiàn)一旦我們只是拿student這個(gè)實(shí)體操作,get方法是會(huì)輸出相關(guān)的sql查詢語句的,而我們注釋掉get方法,換load方法的話,load方法執(zhí)行拿student這個(gè)實(shí)體的數(shù)據(jù)是沒有輸出相應(yīng)的sql語句的。但是我們?cè)趯?duì)student類進(jìn)行操作的時(shí)候,這時(shí)才輸出相關(guān)的語句.
所以我們看出get()和load()之間的差別:
get()每次執(zhí)行都會(huì)執(zhí)行sql語句,不會(huì)延遲。load()方法只是會(huì)在我們對(duì)拿到的實(shí)體進(jìn)行操作的時(shí)候才去執(zhí)行查詢,拿到相應(yīng)的實(shí)體信息。load方法返回的是一個(gè)代理對(duì)象(馬士兵老師視頻)。無論是get還是load首先都會(huì)查一級(jí)緩存(session)中有沒有相應(yīng)的對(duì)應(yīng)值,如果沒有,再去數(shù)據(jù)庫進(jìn)行查找。
clear()方法是對(duì)session里面的緩存進(jìn)行了清理,測試這個(gè)方法我們可以根據(jù)下面的Junit進(jìn)行測試:
@Test public void testClear() { Session session = sessionFactory.getCurrentSession(); session.get(StudentEntity.class,1); session.clear(); session.get(StudentEntity.class,1); }
如果我們把session.clear()注釋掉的話,我們就可以看到只執(zhí)行了一條sql語句
flush()方法是是對(duì)數(shù)據(jù)庫同步的一條語句,執(zhí)行完這個(gè)后立即刷新到數(shù)據(jù)庫。粗淺的來看哈,其實(shí)里面還是有很多東西,比如FlushMode這個(gè)類,我們默認(rèn)設(shè)置的是FlushMode.AUTO,這個(gè)一般不修改,只是為了以后的性能調(diào)優(yōu)。其實(shí)我們?cè)趫?zhí)行事務(wù)的時(shí)候,事務(wù)完成之后它會(huì)幫我們執(zhí)行flush()方法。只是我們?cè)谶M(jìn)行大規(guī)模插入的時(shí)候,我們經(jīng)??梢钥吹较旅娴牟僮鳎?/p>
Session session = sessionFactory.openSession(); Transaction tx = session.beginTransaction(); for ( int i=0; i<100000; i++ ) { Customer customer = new Customer(.....); session.save(customer); if ( i % 20 == 0 ) { //20, same as the JDBC batch size //flush a batch of inserts and release memory: session.flush(); session.clear(); }} tx.commit();session.close();
這樣做的目的就是避免大量插入,造成session里面內(nèi)存溢出,所以我們定時(shí)清理一下就可以避免這個(gè)問題。
以上就是本文的全部內(nèi)容,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來一定的幫助,同時(shí)也希望多多支持億速云!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。