国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 開發(fā) > 綜合 > 正文

Hibernate程序性能優(yōu)化的考慮要點

2024-07-21 02:14:37
字體:
來源:轉載
供稿:網友

  本文依照hibernate幫助文檔,一些網絡書籍及項目經驗整理而成,只提供要點和思路,具體做法可以留言探討,或是找一些更詳細更有針對性的資料。

  初用hibernate的人也許都遇到過性能問題,實現(xiàn)同一功能,用hibernate與用jdbc性能相差十幾倍很正常,如果不及早調整,很可能影響整個項目的進度。

  大體上,對于hibernate性能調優(yōu)的主要考慮點如下:

  ø 數據庫設計調整

  ø hql優(yōu)化

  ø api的正確使用(如根據不同的業(yè)務類型選用不同的集合及查詢api)

  ø 主配置參數(日志,查詢緩存,fetch_size, batch_size等)

  ø 映射文件優(yōu)化(id生成策略,二級緩存,延遲加載,關聯(lián)優(yōu)化)

  ø 一級緩存的管理

  ø 針對二級緩存,還有許多特有的策略

  ø 事務控制策略。

  1、 數據庫設計

  a) 降低關聯(lián)的復雜性

  b) 盡量不使用聯(lián)合主鍵

  c) id的生成機制,不同的數據庫所提供的機制并不完全一樣

  d) 適當的冗余數據,不過分追求高范式

  2、 hql優(yōu)化

  hql如果拋開它同hibernate本身一些緩存機制的關聯(lián),hql的優(yōu)化技巧同普通的sql優(yōu)化技巧一樣,可以很容易在網上找到一些經驗之談。

  3、 主配置

  a) 查詢緩存,同下面講的緩存不太一樣,它是針對hql語句的緩存,即完全一樣的語句再次執(zhí)行時可以利用緩存數據。但是,查詢緩存在一個交易系統(tǒng)(數據變更頻繁,查詢條件相同的機率并不大)中可能會起反作用:它會白白耗費大量的系統(tǒng)資源但卻難以派上用場。

  b) fetch_size,同jdbc的相關參數作用類似,參數并不是越大越好,而應根據業(yè)務特征去設置

  c) batch_size同上。

  d) 生產系統(tǒng)中,切記要關掉sql語句打印。

  4、 緩存

  a) 數據庫級緩存:這級緩存是最高效和安全的,但不同的數據庫可管理的層次并不一樣,比如,在oracle中,可以在建表時指定將整個表置于緩存當中。

  b) session緩存:在一個hibernate session有效,這級緩存的可干預性不強,大多于hibernate自動管理,但它提供清除緩存的方法,這在大批量增加/更新操作是有效的。比如,同時增加十萬條記錄,按常規(guī)方式進行,很可能會發(fā)現(xiàn)outofmemeroy的異常,這時可能需要手動清除這一級緩存:session.evict以及session.clear

  c) 應用緩存:在一個sessionfactory中有效,因此也是優(yōu)化的重中之重,因此,各類策略也考慮的較多,在將數據放入這一級緩存之前,需要考慮一些前提條件:

  i. 數據不會被第三方修改(比如,是否有另一個應用也在修改這些數據?)

  ii. 數據不會太大

  iii. 數據不會頻繁更新(否則使用cache可能適得其反)

  iv. 數據會被頻繁查詢

  v. 數據不是關鍵數據(如涉及錢,安全等方面的問題)。

  緩存有幾種形式,可以在映射文件中配置:read-only(只讀,適用于很少變更的靜態(tài)數據/歷史數據),nonstrict-read-write,read-write(比較普遍的形式,效率一般),transactional(jta中,且支持的緩存產品較少)

  d) 分布式緩存:同c)的配置一樣,只是緩存產品的選用不同,在目前的hibernate中可供選擇的不多,oscache, jboss cache,目前的大多數項目,對它們的用于集群的使用(特別是關鍵交易系統(tǒng))都持保守態(tài)度。在集群環(huán)境中,只利用數據庫級的緩存是最安全的。

  5、 延遲加載

  a) 實體延遲加載:通過使用動態(tài)代理實現(xiàn)

  b) 集合延遲加載:通過實現(xiàn)自有的set/list,hibernate提供了這方面的支持

  c) 屬性延遲加載:

  6、 方法選用

  a) 完成同樣一件事,hibernate提供了可供選擇的一些方式,但具體使用什么方式,可能用性能/代碼都會有影響。顯示,一次返回十萬條記錄(list/set/bag/map等)進行處理,很可能導致內存不夠的問題,而如果用基于游標(scrollableresults)或iterator的結果集,則不存在這樣的問題。

  b) session的load/get方法,前者會使用二級緩存,而后者則不使用。

  c) query和list/iterator,如果去仔細研究一下它們,你可能會發(fā)現(xiàn)很多有意思的情況,二者主要區(qū)別(如果使用了spring,在hibernatetemplate中對應find,iterator方法):

  i. list只能利用查詢緩存(但在交易系統(tǒng)中查詢緩存作用不大),無法利用二級緩存中的單個實體,但list查出的對象會寫入二級緩存,但它一般只生成較少的執(zhí)行sql語句,很多情況就是一條(無關聯(lián))。

  ii. iterator則可以利用二級緩存,對于一條查詢語句,它會先從數據庫中找出所有符合條件的記錄的id,再通過id去緩存找,對于緩存中沒有的記錄,再構造語句從數據庫中查出,因此很容易知道,如果緩存中沒有任何符合條件的記錄,使用iterator會產生n+1條sql語句(n為符合條件的記錄數)

  iii. 通過iterator,配合緩存管理api,在海量數據查詢中可以很好的解決內存問題,如:

  while(it.hasnext()){

  youobject object = (youobject)it.next();

  session.evict(youobject);

  sessionfactory.evice(youobject.class, youobject.getid());

  }

  如果用list方法,很可能就出outofmemory錯誤了。

  iv. 通過上面的說明,我想你應該知道如何去使用這兩個方法了。

  7、 集合的選用

  在hibernate 3.1文檔的“19.5. understanding collection performance”中有詳細的說明。

  8、 事務控制

  事務方面對性能有影響的主要包括:事務方式的選用,事務隔離級別以及鎖的選用

  a) 事務方式選用:如果不涉及多個事務管理器事務的話,不需要使用jta,只有jdbc的事務控制就可以。

  b) 事務隔離級別:參見標準的sql事務隔離級別

  c) 鎖的選用:悲觀鎖(一般由具體的事務管理器實現(xiàn)),對于長事務效率低,但安全。樂觀鎖(一般在應用級別實現(xiàn)),如在hibernate中可以定義version字段,顯然,如果有多個應用操作數據,且這些應用不是用同一種樂觀鎖機制,則樂觀鎖會失效。因此,針對不同的數據應有不同的策略,同前面許多情況一樣,很多時候我們是在效率與安全/準確性上找一個平衡點,無論如何,優(yōu)化都不是一個純技術的問題,你應該對你的應用和業(yè)務特征有足夠的了解。

  9、 批量操作

  即使是使用jdbc,在進行大批數據更新時,batch與不使用batch有效率上也有很大的差別。我們可以通過設置batch_size來讓其支持批量操作。

  舉個例子,要批量刪除某表中的對象,如“delete account”,打出來的語句,會發(fā)現(xiàn)hibernate找出了所有account的id,再進行刪除,這主要是為了維護二級緩存,這樣效率肯定高不了,在后續(xù)的版本中增加了bulk delete/update,但這也無法解決緩存的維護問題。也就是說,由于有了二級緩存的維護問題,hibernate的批量操作效率并不盡如人意!

  從前面許多要點可以看出,很多時候我們是在效率與安全/準確性上找一個平衡點,無論如何,優(yōu)化都不是一個純技術的問題,你應該對你的應用和業(yè)務特征有足夠的了解,一般的,優(yōu)化方案應在架構設計期就基本確定,否則可能導致沒必要的返工,致使項目延期,而作為架構師和項目經理,還要面對開發(fā)人員可能的抱怨,必竟,我們對用戶需求更改的控制力不大,但技術/架構風險是應該在初期意識到并制定好相關的對策。

  還有一點要注意,應用層的緩存只是錦上添花,永遠不要把它當救命稻草,應用的根基(數據庫設計,算法,高效的操作語句,恰當api的選擇等)才是最重要的。

  • 本文來源于網頁設計愛好者web開發(fā)社區(qū)http://www.html.org.cn收集整理,歡迎訪問。
  • 發(fā)表評論 共有條評論
    用戶名: 密碼:
    驗證碼: 匿名發(fā)表
    主站蜘蛛池模板: 鸡西市| 巨鹿县| 榆树市| 鄯善县| 嘉祥县| 达拉特旗| 深州市| 大理市| 扬中市| 乐陵市| 武平县| 临江市| 宁安市| 江口县| 通化县| 嘉祥县| 水城县| 西丰县| 陆良县| 博野县| 长乐市| 新乡县| 含山县| 崇义县| 平阳县| 阜南县| 浙江省| 甘谷县| 洞口县| 府谷县| 怀集县| 淳安县| 名山县| 宁蒗| 西畴县| 临猗县| 扎兰屯市| 章丘市| 扎兰屯市| 多伦县| 吐鲁番市|