![]() |
Модераторы: LSD, AntonSaburov |
![]() ![]() ![]() |
|
LuMee |
|
|||
Шустрый ![]() Профиль Группа: Участник Сообщений: 117 Регистрация: 30.3.2007 Репутация: нет Всего: 1 |
Много где видел похожие вопросы, но так и не нашел внятного ответа.
Итак, пусть имеются две сущности, скажем, тема форума и сообщение в теме:
Вопросы таковы: 1. можно ли вытащить с помощью EntityManager'а объект Topic, не извлекая для него список Post'ов? А то, если уж говорить о форуме, то для извлечения списка топиков для отображения его на главной странице совершенно необязательно тащить еще и сообщения (так никакой памяти не хватит) 2. если удался п. 1, то как надежно проверить, что коллекция posts не проинициализирована? Она будет просто null, или там что похитрее? 3. опять же, если все нормально прошло с п.1, то как потом при необходимости все-таки проинициализировать эту коллекцию (тут уже догадки есть, но пока нет возможности проверить)? По п.1, знаю, есть у аннотации OneToMany параметр fetchType, но, как я понял, установка ему значения FetchType.LAZY не гарантирует, что дочерние сущности не будут извлечены, ибо сервер приложений может данный параметр просто проигнорировать. Есть ли надежный способ убедить его лишние сущности не тащить, или остается только положиться на его умение следить за ресурсами? |
|||
|
||||
Stampede |
|
|||
![]() Гносеолог ![]() ![]() Профиль Группа: Участник Клуба Сообщений: 963 Регистрация: 25.4.2005 Где: Calgary, Alberta, Canada Репутация: 66 Всего: 144 |
Коль скоро речь идет о веб приложении (форум), то я бы вообще не полагался на автоматический подгруз связанных сущностей. Дело в том, что в вебных интерфейсах практически для любых видов коллекций нужно предусматривать постраничный просмотр. И это требование делает объявления вида @OneToMany практически бесполезными.
-------------------- "If you want something done right, do it yourself" По секрету: выучить английский - реально! |
|||
|
||||
LuMee |
|
|||
Шустрый ![]() Профиль Группа: Участник Сообщений: 117 Регистрация: 30.3.2007 Репутация: нет Всего: 1 |
Мда, неудачно я видимо пример привел... А если рассматривать более общую ситуацию, нежели сущности для отображения на странице? Пусть, скажем, это будут некие сущности, описывающие какие-то внутренности системы или еще чего. У меня пока складывается впечатление, что самым простым и надежным способом добиться желаемого результата будет вообще забыть про OneToMany и вытаскивать все данные самостоятельно, однако это уже не так красиво.
|
|||
|
||||
ecologist |
|
|||
![]() Опытный ![]() ![]() Профиль Группа: Участник Сообщений: 452 Регистрация: 28.2.2007 Репутация: 5 Всего: 9 |
По моему мнению EJB использует идеологию Hibernate, а там LAZY будет нормально работать только в рамках транзакции. Отсюда у меня вывод:
Использовать "ленивые" коллекции можно и нужно, но каждый раз это должно продумываться и тестироваться. |
|||
|
||||
LuMee |
|
|||
Шустрый ![]() Профиль Группа: Участник Сообщений: 117 Регистрация: 30.3.2007 Репутация: нет Всего: 1 |
Оно понятно, что можно и нужно (не зря же люди их делали
![]() По моему мнению EJB использует идеологию Hibernate, а там LAZY будет нормально работать только в рамках транзакции Вот с этого места можно немного поподробнее? А то с Hibernate лично дел не имел. |
|||
|
||||
Maverick |
|
|||
![]() Эксперт ![]() ![]() ![]() Профиль Группа: Завсегдатай Сообщений: 1307 Регистрация: 22.9.2003 Где: Odessa, Ukraine Репутация: 2 Всего: 10 |
Как все-таки проинициализировать коллекцию... ?
![]() |
|||
|
||||
AntonSaburov |
|
|||
![]() Штурман ![]() ![]() ![]() ![]() Профиль Группа: Модератор Сообщений: 5658 Регистрация: 2.7.2002 Где: Санкт-Петербург Репутация: 8 Всего: 118 |
Строчку с session - там получение сессии Hibernate. А дальше - первый шаг - получение объекта Agency. И после инициализация двух коллекций - терминалы и мнемоники. Ну это просто кусок выдраный из кода. Вот что важно - Hibernate.initialize(...) |
|||
|
||||
Maverick |
|
|||
![]() Эксперт ![]() ![]() ![]() Профиль Группа: Завсегдатай Сообщений: 1307 Регистрация: 22.9.2003 Где: Odessa, Ukraine Репутация: 2 Всего: 10 |
А в Ejb3?
Добавлено через 2 минуты и 36 секунд у меня вот такую хрень выдает
если прописать fetchtype =EAGER, то возникает другая.... ну не хочу я пока использовать эту ленивую загрузку, чего она лезет... |
|||
|
||||
AntonSaburov |
|
|||
![]() Штурман ![]() ![]() ![]() ![]() Профиль Группа: Модератор Сообщений: 5658 Регистрация: 2.7.2002 Где: Санкт-Петербург Репутация: 8 Всего: 118 |
Видимо ты пытаешься инициализировать коллекцию уже за границей открытия сессии.
По сути ты можешь вытащить коллекцию ТОЛЬКО пока сессия еще открыта. Как только ты ее закрываешь, то все. Образно - открыли дверку (сессию) - и ты можешь таскать данные, записывать и прочая. Как только дверка закрыта (сессия) - все. Тебе так и написали. |
|||
|
||||
Maverick |
|
|||
![]() Эксперт ![]() ![]() ![]() Профиль Группа: Завсегдатай Сообщений: 1307 Регистрация: 22.9.2003 Где: Odessa, Ukraine Репутация: 2 Всего: 10 |
Дело в том, что я не использую прямо сессии и ленивую загрузку... я вообще не хочу это делать прямо - не настолько опытен еще.... а все равно вылазит всякое...
Добавлено через 1 минуту и 45 секунд
если можно - чуть подробнее об этой строке.... что такое template? Это экземпляр чего? Добавлено через 2 минуты и 13 секунд И в каком методе сидят эти строчки? |
|||
|
||||
Maverick |
|
|||
![]() Эксперт ![]() ![]() ![]() Профиль Группа: Завсегдатай Сообщений: 1307 Регистрация: 22.9.2003 Где: Odessa, Ukraine Репутация: 2 Всего: 10 |
Ставлю fetch = EAGER вот такая штука вылазит
|
|||
|
||||
Maverick |
|
|||
![]() Эксперт ![]() ![]() ![]() Профиль Группа: Завсегдатай Сообщений: 1307 Регистрация: 22.9.2003 Где: Odessa, Ukraine Репутация: 2 Всего: 10 |
нельзя ли расписать примерчик... очень прошу - кто уже разобрался...
есть два entity со связь @OneToMany, даже три - вложенных... вот так примерно...
как с ними обращаться, чтобы все коллекции отвечали нормально? где-какие сессии открывать? как их проинициалировать? я пытаюсь и так и этак... пол инета перерыл... Добавлено через 49 секунд хотя бы направьте - в каком направлении рыть... заранее благодарен... |
|||
|
||||
Maverick |
|
|||
![]() Эксперт ![]() ![]() ![]() Профиль Группа: Завсегдатай Сообщений: 1307 Регистрация: 22.9.2003 Где: Odessa, Ukraine Репутация: 2 Всего: 10 |
В последнее время -у меня стойкое мнение, что я один русскоязычный на планете занимаюсь EJB3...
проблема описана ... но ее решение в данном случае мне не подходит - там чистый Hibernate... да оно и не работает нормально под EJB3... |
|||
|
||||
Maverick |
|
|||
![]() Эксперт ![]() ![]() ![]() Профиль Группа: Завсегдатай Сообщений: 1307 Регистрация: 22.9.2003 Где: Odessa, Ukraine Репутация: 2 Всего: 10 |
Переключил проект на TopLink - все заработало сразу и нормально...
![]() |
|||
|
||||
AntonSaburov |
|
||||||
![]() Штурман ![]() ![]() ![]() ![]() Профиль Группа: Модератор Сообщений: 5658 Регистрация: 2.7.2002 Где: Санкт-Петербург Репутация: 8 Всего: 118 |
Эту строчку как раз не надо рассматривать - это наш внутренний класс - но там вообщем надо получить сессию. Вот из той же доки по Hibernate
|
||||||
|
|||||||
Maverick |
|
|||
![]() Эксперт ![]() ![]() ![]() Профиль Группа: Завсегдатай Сообщений: 1307 Регистрация: 22.9.2003 Где: Odessa, Ukraine Репутация: 2 Всего: 10 |
и где это надо использовать? в сессионном фасаде? или в самом ентити? как же инициализировать коллекцию все-таки? где именно это надо сделать?
AntonSaburov, раскажите хотя бы кратко как это сделать и вы сделаете этот топик одним из самых популярных в интернете... Вы пользуетесь EJB3 в работе? |
|||
|
||||
AntonSaburov |
|
|||
![]() Штурман ![]() ![]() ![]() ![]() Профиль Группа: Модератор Сообщений: 5658 Регистрация: 2.7.2002 Где: Санкт-Петербург Репутация: 8 Всего: 118 |
Ох, в реальном проекте у нас к сожалению используется еще EJB 2 и Hibernate (убедили начальство, что это проще). А вот для "Студенческого отдела кадров" я копался. Но вот описать прямо с нуля - сложно. В NetBeans это делается весьма удобно - наверно проще будет просто описать действия по пунктам. Постараюсь что-то простое выложить. А дальше уже самими копаться придется ![]() |
|||
|
||||
Maverick |
|
|||
![]() Эксперт ![]() ![]() ![]() Профиль Группа: Завсегдатай Сообщений: 1307 Регистрация: 22.9.2003 Где: Odessa, Ukraine Репутация: 2 Всего: 10 |
AntonSaburov, я копаюсь со страшной силой... сижу как раз в NetBeans - действительно удобно проектировать и компелировать, и деплоить.... но, не работает проект сделанный теоретически полностью правильно...
Добавлено через 2 минуты и 4 секунды Как можно было сделать библиотеку Hibernate так, чтобы в одном классе нельзя было хранить две подгруженные постоянно коллекции? У меня в голове не укладывается... ведь это есть в каждом втором проекте... |
|||
|
||||
AlexeyVorotnikov |
|
|||
Опытный ![]() ![]() Профиль Группа: Участник Сообщений: 658 Регистрация: 18.6.2007 Где: Москва Репутация: 3 Всего: 18 |
Это в каком смысле? -------------------- RTFM! Три источника и три составные части Java: The Java Language Specification, Java Platform API Specification, The Java Virtual Machine Specification |
|||
|
||||
Maverick |
|
|||
![]() Эксперт ![]() ![]() ![]() Профиль Группа: Завсегдатай Сообщений: 1307 Регистрация: 22.9.2003 Где: Odessa, Ukraine Репутация: 2 Всего: 10 |
В том смысле, что если скажем у тебя есть экономический агент, у которого есть список деклараций и список субсчетов одновременно - то стандартно их проиницилизировать через Hibernate в EJB3 не получиться....
Добавлено через 29 секунд да и вообще непонятно - как инициализировать их... |
|||
|
||||
hamsterKSU |
|
|||
Опытный ![]() ![]() Профиль Группа: Участник Сообщений: 401 Регистрация: 20.10.2006 Где: Украина, Херсон Репутация: 3 Всего: 11 |
может проблема и решена я так и непонял но насчет инициализации колекций меня в Hibernate спасло следующие
@Fetch(FetchMode.SUBSELECT) @OneToMany(mappedBy = "...", fetch = FetchType.EAGER) |
|||
|
||||
Maverick |
|
|||
![]() Эксперт ![]() ![]() ![]() Профиль Группа: Завсегдатай Сообщений: 1307 Регистрация: 22.9.2003 Где: Odessa, Ukraine Репутация: 2 Всего: 10 |
К сожалению не помогло... проблема остается открытой...
|
|||
|
||||
Maverick |
|
|||
![]() Эксперт ![]() ![]() ![]() Профиль Группа: Завсегдатай Сообщений: 1307 Регистрация: 22.9.2003 Где: Odessa, Ukraine Репутация: 2 Всего: 10 |
хух... заработало... решение проблемы здесь - в этом блоге...
единственным недостатком является то, что приходиться отказаться от чистого EJB3 и использовать расширение Hibernate @IndexColumn... однако, странновато.... EJB3 сыроват... ![]() |
|||
|
||||
VSergeyV |
|
|||
Опытный ![]() ![]() Профиль Группа: Участник Сообщений: 299 Регистрация: 9.5.2006 Где: Новосибирск Репутация: нет Всего: 3 |
Как можно повлиять на связанные сущности при выборке?
Пример топискатера
Допустим выбираем Topic по id, а в posts нужно выбрать не просто все посты для темы с заданным id (что делается по умолчанию), но и еще со status="А". |
|||
|
||||
victorq10 |
|
||||||||||||||||
Новичок Профиль Группа: Участник Сообщений: 3 Регистрация: 5.8.2009 Где: Київ Репутация: нет Всего: нет |
Если еще кто зайдет на эту тему, то мой совет такой.
Осоциации для бинов OneToMany и ManyToOne служат для того чтобы можно было пользоваться JPQL. Пример: Выбираем топики со статусом, к примеру, 'A'
or
or
Выбираем все посты где у топиков type = 1
or
or
Все запросы должны возвращать одно и тоже. Последний запрос (без JOIN) может использоваться и без объявленных связей OneToMany и/или ManyToOne у соответствующих объектах. Заключение: OneToMany, ManyToOne, OneToOne, ManyToMany описывают поля для того, чтобы было удобно строить запросы на JPQL. JPA делает важную вещь - возвращает объекты!!! Не кто не мешает по ассоциации OneToMany получить полную коллекцию связанных объектов и в цикле отобрать те что нужно, но такое решение при больших коллекциях, может дольше работать, но при достаточной мощности сервера, это можно не заметить. Когда уже сильно будет тормозить, можно оптимизировать ![]() ![]() ![]() У меня волнует другой вопрос, с которым я столкнулся. как избежать циклическую загрузки по связям ManyToOne. К примеру выбираем "Клиентов" с базы, у клиентов есть "Менеджер", он в свою очередь принадлежит к "ГрупеМенеджеров", так же у клиектов имеет "СтатусКлиента". В результате будет один запрос с таблици "Клиентов" и много одиночных запросов к таблицам: "Менеджер", "ГрупеМенеджеров", "СтатусКлиента". если 1000 клиентов, 120 СтатусовКлиента, 40 менеджеров, 5 ГрупМенеджеров, и все записи из таблиц будут использоваться, то получим минимум 1 + 120 + 40 + 5 = 167 запросов. TopLink сделает где-то 2000 запросов (у меня чуть другие данные, но суть та же) Нашел ответ что нужно проектировать структуру базы с учетом связей. Как бы это хорошо, но может есть более лояльный подход. Другой вариант оптимизации использовать выбрав только то что нужно
Третий вариант оптимизации использовать LEFT JOIN FETCH:
этот вариант у меня не заработал в TopLink, в других не пробовал (пример для Hibernate). у меня работает только один уровень с FETCH. (число одиночных запросов уменьшилось ![]() Четвертый вариант использовать FetchGroup, не совсем понял как его использовать и он зависит от реализации JPA. Что делать в такой ситуации? Как избежать циклических запросов по связям ManyToOne когда в базе присутствуют сложные зависимости? |
||||||||||||||||
|
|||||||||||||||||
![]() ![]() ![]() |
Правила форума "Java" | |
|
Если Вам помогли, и атмосфера форума Вам понравилась, то заходите к нам чаще! С уважением, LSD, AntonSaburov, powerOn, tux. |
0 Пользователей читают эту тему (0 Гостей и 0 Скрытых Пользователей) | |
0 Пользователей: | |
« Предыдущая тема | Java EE (J2EE) и Spring | Следующая тема » |
|
По вопросам размещения рекламы пишите на vladimir(sobaka)vingrad.ru
Отказ от ответственности Powered by Invision Power Board(R) 1.3 © 2003 IPS, Inc. |