Hibernate中g(shù)et和load有什么不同之處?
把get和load放到一起進(jìn)行對比是Hibernate面試時(shí)最常問到的問題,這是因?yàn)橹挥姓_理解get()和load()這二者后才有可能高效地使用Hibernate。get和load的最大區(qū)別是,如果在緩存中沒有找到相應(yīng)的對象,get將會(huì)直接訪問數(shù)據(jù)庫并返回一個(gè)完全初始化好的對象,而這個(gè)過程有可能會(huì)涉及到多個(gè)數(shù)據(jù)庫調(diào)用;而load方法在緩存中沒有發(fā)現(xiàn)對象的情況下,只會(huì)返回一個(gè)代理對象,只有在對象getId()之外的其它方法被調(diào)用時(shí)才會(huì)真正去訪問數(shù)據(jù)庫,這樣就能在某些情況下大幅度提高性能。
Hibernate中save、persist和saveOrUpdate這三個(gè)方法的不同之處?
除了get和load,這又是另外一個(gè)經(jīng)常出現(xiàn)的Hibernate面試問題。 所有這三個(gè)方法,也就是save()、saveOrUpdate()和persist()都是用于將對象保存到數(shù)據(jù)庫中的方法,但其中有些細(xì)微的差別。例如,save()只能INSERT記錄,但是saveOrUpdate()可以進(jìn)行 記錄的INSERT和UPDATE。還有,save()的返回值是一個(gè)Serializable對象,而persist()方法返回值為void。
Hibernate中的命名SQL查詢指的是什么?
Hibernate的這個(gè)面試問題同Hibernate提供的查詢功能相關(guān)。命名查詢指的是用
Hibernate中的SessionFactory有什么作用? SessionFactory是線程安全的嗎?
這也是Hibernate框架的常見面試問題。顧名思義,SessionFactory就是一個(gè)用于創(chuàng)建Hibernate的Session對象的工廠。SessionFactory通常是在應(yīng)用啟動(dòng)時(shí)創(chuàng)建好的,應(yīng)用程序中的代碼用它來獲得Session對象。作為一個(gè)單個(gè)的數(shù)據(jù)存儲(chǔ),它也是 線程安全的,所以多個(gè)線程可同時(shí)使用同一個(gè)SessionFactory。Java JEE應(yīng)用一般只有一個(gè)SessionFactory,服務(wù)于客戶請求的各線程都通過這個(gè)工廠來獲得Hibernate的Session實(shí)例,這也是為什么SessionFactory接口的實(shí)現(xiàn)必須是線程安全的原因。還有,SessionFactory的內(nèi)部狀態(tài)包含著同對象關(guān)系影射有關(guān)的所有元數(shù)據(jù),它是 不可變的,一旦創(chuàng)建好后就不能對其進(jìn)行修改了。
Hibernate中的Session指的是什么? 可否將單個(gè)的Session在多個(gè)線程間進(jìn)行共享?
前面的問題問完之后,通常就會(huì)接著再問這兩個(gè)問題。問完SessionFactory的問題后就該輪到Session了。Session代表著Hibernate所做的一小部分工作,它負(fù)責(zé)維護(hù)者同數(shù)據(jù)庫的鏈接而且 不是線程安全的,也就是說,Hibernage中的Session不能在多個(gè)線程間進(jìn)行共享。雖然Session會(huì)以主動(dòng)滯后的方式獲得數(shù)據(jù)庫連接,但是Session最好還是在用完之后立即將其關(guān)閉。
hibernate中sorted collection和ordered collection有什么不同?
T這個(gè)是你會(huì)碰到的所有Hibernate面試問題中比較容易的問題。sorted collection是通過使用 Java的Comparator在內(nèi)存中進(jìn)行排序的,ordered collection中的排序用的是數(shù)據(jù)庫的order by子句。對于比較大的數(shù)據(jù)集,為了避免在內(nèi)存中對它們進(jìn)行排序而出現(xiàn) Java中的OutOfMemoryError,最好使用ordered collection。
Hibernate中transient、persistent、detached對象三者之間有什么區(qū)別?
在Hibernate中,對象具有三種狀態(tài):transient、persistent和detached。同Hibernate的session有關(guān)聯(lián)的對象是persistent對象。對這種對象進(jìn)行的所有修改都會(huì)按照事先設(shè)定的刷新策略,反映到數(shù)據(jù)庫之中,也即,可以在對象的任何一個(gè)屬性發(fā)生改變時(shí)自動(dòng)刷新,也可以通過調(diào)用Session.flush()方法顯式地進(jìn)行刷新。如果一個(gè)對象原來同Session有關(guān)聯(lián)關(guān)系,但當(dāng)下卻沒有關(guān)聯(lián)關(guān)系了,這樣的對象就是detached的對象。你可以通過調(diào)用任意一個(gè)session的update()或者saveOrUpdate()方法,重新將該detached對象同相應(yīng)的seesion建立關(guān)聯(lián)關(guān)系。Transient對象指的是新建的持久化類的實(shí)例,它還從未同Hibernate的任何Session有過關(guān)聯(lián)關(guān)系。同樣的,你可以調(diào)用persist()或者save()方法,將transient對象變成persistent對象。可要記住,這里所說的transient指的可不是 Java中的transient關(guān)鍵字,二者風(fēng)馬牛不相及。
Hibernate中Session的lock()方法有什么作用?
這是一個(gè)比較棘手的Hibernate面試問題,因?yàn)镾ession的lock()方法重建了關(guān)聯(lián)關(guān)系卻并沒有同數(shù)據(jù)庫進(jìn)行同步和更新。因此,你在使用lock()方法時(shí)一定要多加小心。順便說一下,在進(jìn)行關(guān)聯(lián)關(guān)系重建時(shí),你可以隨時(shí)使用Session的update()方法同數(shù)據(jù)庫進(jìn)行同步。有時(shí)這個(gè)問題也可以這么來問:Session的lock()方法和update()方法之間有什么區(qū)別?。這個(gè)小節(jié)中的關(guān)鍵點(diǎn)也可以拿來回答這個(gè)問題。
Hibernate中二級緩存指的是什么?
這是同Hibernate的緩存機(jī)制相關(guān)的第一個(gè)面試問題,不出意外后面還會(huì)有更多這方面的問題。二級緩存是在SessionFactory這個(gè)級別維護(hù)的緩存,它能夠通過節(jié)省幾番數(shù)據(jù)庫調(diào)用往返來提高性能。還有一點(diǎn)值得注意,二級緩存是針對整個(gè)應(yīng)用而不是某個(gè)特定的session的。
為什么在Hibernate的實(shí)體類中要提供一個(gè)無參數(shù)的構(gòu)造器這一點(diǎn)非常重要?
每個(gè)Hibernate實(shí)體類必須包含一個(gè) 無參數(shù)的構(gòu)造器, 這是因?yàn)镠ibernate框架要使用Reflection API,通過調(diào)用Class.newInstance()來創(chuàng)建這些實(shí)體類的實(shí)例。如果在實(shí)體類中找不到無參數(shù)的構(gòu)造器,這個(gè)方法就會(huì)拋出一個(gè)InstantiationException異常。
可不可以將Hibernate的實(shí)體類定義為final類?
是的,你可以將Hibernate的實(shí)體類定義為final類,但這種做法并不好。因?yàn)镠ibernate會(huì)使用代理模式在延遲關(guān)聯(lián)的情況下提高性能,如果你把實(shí)體類定義成final類之后,因?yàn)?Java不允許對final類進(jìn)行擴(kuò)展,所以Hibernate就無法再使用代理了,如此一來就限制了使用可以提升性能的手段。不過,如果你的持久化類實(shí)現(xiàn)了一個(gè)接口而且在該接口中聲明了所有定義于實(shí)體類中的所有public的方法輪到話,你就能夠避免出現(xiàn)前面所說的不利后果。
相關(guān)導(dǎo)讀: