看板 java 關於我們 聯絡資訊
最近利用空檔在練習 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