не удалось лениво инициализировать коллекцию ролей - PullRequest
27 голосов
/ 23 марта 2011

Привет У меня есть два класса, как это:

public class Indicator implements Serializable {
...

    @OneToMany(mappedBy = "indicator",fetch=FetchType.LAZY)
    private List<IndicatorAlternateLabel>  indicatorAlternateLabels;

    public List<IndicatorAlternateLabel> getIndicatorAlternateLabels() {
        return indicatorAlternateLabels;
    }

        public void setIndicatorAlternateLabels(List<IndicatorAlternateLabel> indicatorAlternateLabels) {
            this.indicatorAlternateLabels = indicatorAlternateLabels;
        }
...
    }

public class IndicatorAlternateLabel implements Serializable {
...
 @ManyToOne(cascade = CascadeType.REFRESH, fetch = FetchType.EAGER)
    @JoinColumn(name = "IndicatorID")
    @XmlTransient
    private Indicator indicator;
...
}

Когда я использую их следующим образом:

 public MetricTypeDetail getMetricTypeDetail(Integer metricTypeId) {
        Criteria crit = sessionFactory.getCurrentSession().createCriteria(Indicator.class, "sub")
                    .add(Restrictions.eq("number", metricTypeId))
                    .setResultTransformer(Criteria.DISTINCT_ROOT_ENTITY).setCacheable(true);
        crit.setMaxResults(1);
        Indicator indicator=(Indicator) crit.uniqueResult();
        MetricTypeDetail metricTypeDetail=new MetricTypeDetail(indicator);
        List<IndicatorAlternateLabel> indicatorAlternateLabels = null;
        indicatorAlternateLabels=indicator.getIndicatorAlternateLabels();
        metricTypeDetail.setIndicatorAlternateLabels(indicatorAlternateLabels);
        return metricTypeDetail;
    }

Этот код возвращает исключение: не удалось лениво инициализировать коллекцию ролей: com.porism.service.domain.Indicator.indicatorAlternateLabels, ни один сеанс или сеанс не был закрыт

Есть идеи?Я очень плохо знаком с Hibernate

Ответы [ 3 ]

76 голосов
/ 23 марта 2011

Ленивые исключения возникают, когда вы выбираете объект, обычно содержащий коллекцию, которая загружается лениво, и пытаетесь получить доступ к этой коллекции.

Вы можете избежать этой проблемы,

  • доступ к ленивой коллекции в транзакции.
  • Инициализация коллекции с использованием Hibernate.initialize(obj);
  • Выбрать коллекцию в другой транзакции
  • Используйте Fetch profiles для выбора времени ленивого / не ленивого извлечения
  • Установить выборку на ленивый (что обычно не рекомендуется)

Далее я бы порекомендовал посмотреть ссылки related справа от вас, где на этот вопрос уже много раз отвечали. Также см. Дизайн приложения для отложенной загрузки в спящем режиме .

3 голосов
/ 25 июня 2012

Возможно, вы не получаете присоединенный набор.Обязательно включите набор в свой HQL:

public List<Node> getAll() {
    Session session = sessionFactory.getCurrentSession();
    Query query = session.createQuery("FROM Node as n LEFT JOIN FETCH n.nodeValues LEFT JOIN FETCH n.nodeStats");
    return  query.list();
}

, где у вашего класса есть 2 набора, например:

public class Node implements Serializable {

@OneToMany(fetch=FetchType.LAZY)
private Set<NodeValue> nodeValues;

@OneToMany(fetch=FetchType.LAZY)
private Set<NodeStat> nodeStats;

}
1 голос
/ 23 января 2016

Попробуйте swich fetchType от LAZY до EAGER

...
@OneToMany(fetch=FetchType.EAGER)
private Set<NodeValue> nodeValues;
...

Но в этом случае ваше приложение все равно будет получать данные из БД. Если этот запрос очень сложный - это может повлиять на производительность. Больше здесь: https://docs.oracle.com/javaee/6/api/javax/persistence/FetchType.html

==> 73

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...