作者cyclone350 (老子我最神)
看板java
標題[問題] hibernate 新手問題
時間Wed Jan 9 00:08:40 2013
最近利用空檔在練習 hibernate (進度緩慢)
因為這問題想了很久,在網路上也不知道關鍵字是甚麼,看了一些 document 也沒頭緒
所以就上來請教各位前輩
我使用的是 spring + hibernate
首先 hibernate 是一個 ORM framework。
如果我們要 select,利用他提供的 libery 就可以直接存取。
要查詢物件也很簡單,例如:
String hql = "from people";
List<people> peoplesss = this.getSessionTemplete().creatQuery(hql).list();
就可以得到這些Entity,相反的 insert 跟 delete 也類似
問題來啦,假設這是美食獵人世界的系統資料庫,如果只是用 from people,
則這個 List 的 size 可能好幾百億,這下不得了。
不過 hibernate 可以設定 pageSize 跟 pageNum 之類的東西,可以設定一次只拿到 n 筆
問題是 spring 提供的 hibernateTemplete 並沒有這個功能‧‧‧
好家在 google 大神跟我說 hibernateTemplete 有一個 callbackHibernate 方法,
所以此問題就解決了。
接下來要設定關聯資料庫
假設美食世界的人有三種,美食獵人,廚師,一般人,
我們可以增加一個 Table 名為 FOOD_TAG。
並且在 PEOPLE 的 table 增加一個 FK 名為 TAG,對應到 FOOD_TAG 的 PK,
所以我們在 PEOPLE 的 Entity 要設定一個 @ManyToOne 對應到 FOOD_TAG
而 FOOD_TAG 的 Entity 要設定一個 @OneToMany 對應到 PEOPLE。
這樣想要知道屬於美食獵人Tag的人有哪些,只需要從 FOOD_TAG 的 Entity 拿到 people
的 list 就好了。
問題來了,如果我只需要拿到 Tag 的名稱,例如:
Tag tag = getHibernateTemplete().
createQuery("from tag t where t.name = '一般人'");
String tagName = tag.getName();
那這樣還需要抓取 people List 嗎?
還好可以設定 fetch lazy 來解決這問題,我可以等到要 getPeopleList 時才 query
(不過 session 不能關啦,或是要設定其他的)
不過又有問題了,假設我要 getPeopleList,他會一次給我全部的人,
假設分類在 "一般人" 的 People 有 60 億人,那我的 memory 豈不就爆掉了。
於是我翻書,請 google 搜尋此類問題,皆找不到 OneToMany 的限制用法
我想要 OneToMany 可以想一般平常的 query 可以設定 maxSize 或是 page 的概念
不過 google 似乎也沒找到 ...
另外本來想用 FK 當 where 條件當 query,然後就用一開始的方法。
可是在 entity 的 FK 是隱藏的 (joinColumn)。
想請問有沒有此類方法可以解? Annotation 要怎麼設定才會有這樣的效果
看過很多網站都沒講這問題
有看到一個網頁有用 @Size(max=30) 這樣,但是我的專案根本沒有 @Size
另外還有一個問題,我還沒自己試過,請問 Entity 的 @oneToMany 之類的會有
連鎖效應嗎?
例如 A -> B -> C -> D <- A
A 對 B , 一對多
B 對 C , 一對多
C 對 D , 一對多
A 對 D , 一對多
假設我都是 fetch eager 的話
會在 query A 的時候,把 B 一起 query。
因為 B -> C,hibernate 會順便 query C 嗎?
然後 C -> D 再 query D ?
假設我要拿到D
就直接 a.getB().get(0).getC().get(0).getD() ?
這樣找一個資料,數量級大一點的話,hibernate 不就死掉了嗎?
文有點長,看不董跟我說我再解釋一下,文筆有點不順。
--
※ 發信站: 批踢踢實業坊(ptt.cc)
◆ From: 114.198.167.187
推 LaPass:翻過hibernate,但也沒實務經驗,我也想問這些問題.... 01/09 00:21
推 joetsai:ORM的Paging應該都是用Memory 這樣不會爆掉嗎!? 01/09 00:25
推 phstudy:@LazyCollection(LazyCollectionOption.EXTRA) 01/09 01:34
→ phstudy:JPA2據我所知應沒有,但可用JPQL或Criteria自己寫 01/09 01:37
推 phstudy:fetch eager理論上會全載入,你可以設定show_sql驗證 01/09 01:40
推 swpoker:怎麼會想要一次拿到全部的資料呢? 01/09 09:41
→ cyclone350:就是希望不要一次拿全部的資料,但是在 hibernate 01/09 09:53
→ cyclone350:OneToMany 找不到限制資料量的方法或設定 01/09 09:54
推 swpoker:HIBERNATE想要跨資料庫平台~可惜每家廠商都不太一樣 01/09 10:38
→ swpoker:自己下SQL或許會好點~也要小心DB的負荷~AP跟DB都要注意 01/09 10:41
推 LaPass:我有另外一個問題想問..... 萬一有使用叢集,多個伺服器使 01/09 10:45
→ LaPass:用同個資料庫,會有問題嗎? (hibernate好像會先存在記憶?) 01/09 10:47
→ swpoker:HIBERNET有內坎在DB裡面嗎?(誤) ~ 還是要看DB拉 XD 01/09 11:47
→ cyclone350:結論是在 Hibernate 內無解嗎? 我知道可以用 JPQL 或是 01/09 18:04
→ cyclone350:PL/SQL 或 Criteria,不過感覺是在避開問題,這樣 01/09 18:12
→ cyclone350:@OneToMany 不就形同虛設了嗎? 01/09 18:14
推 Curapikt:fetch eager是最下策,hibernate有個design pattern叫 01/10 01:09
推 Curapikt:openSessionInView ;OneToMany的意義也不是讓你只抓少 01/10 01:12
推 Curapikt:數幾個Entity,你看文件時他都會寫說那些方法不建議使用 01/10 01:14
推 Curapikt:你要抓少數用查詢做就好啦... 01/10 01:17
→ swpoker:其實我覺得你想的太過發散了~不過這個是好事~ 01/10 11:38
→ swpoker:建議你寫出一下整個流程~重新整理思緒~會很有幫助的 01/10 11:40