溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Hibernate三種狀態(tài)和Session常用的方法

發(fā)布時(shí)間:2020-08-25 20:30:52 來源:腳本之家 閱讀:116 作者:尋找現(xiàn)實(shí)扭曲力場 欄目:編程語言

我們知道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的。

Hibernate三種狀態(tài)和Session常用的方法

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í)也希望多多支持億速云!

向AI問一下細(xì)節(jié)

免責(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)容。

AI