Исключение отложенной загрузки при использовании JSF Converter (ссылка на коллекцию) - PullRequest
3 голосов
/ 27 марта 2012

Это мой первый пост после многих исследований этой проблемы.

Этот пример работает под Jboss 7.1 со швом 3.1 (припой + постоянство + грани) с контекстом управляемого шва

Я столкнулся с проблемой, классической failed to lazily initialize a collection, no session or session was closed: org.hibernate.LazyInitializationException: failed to lazily initialize a collection, no session or session was closed при использовании конвертера на бинах Entity. Цель состоит в том, чтобы оставаться на 100% объектно-ориентированным, используя модель JPA.

в beans.xml, org.jboss.seam.transaction.TransactionInterceptor активировано

Сущности:

@Entity
public class Member implements Serializable {

    @Id
    @GeneratedValue
    private Long id;
    private String name;
    private String email;

    @Column(name = "phone_number")
    private String phoneNumber;

    @ManyToMany
    private List<Statut> listeStatut = new ArrayList<Statut>();

    // getters, setters, hashcode, equals
}

@Entity
public class Statut implements Serializable {

    @Id
    @GeneratedValue
    private Long id;

    private String name;

    @ManyToMany(mappedBy="listeStatut")
    private List<Member> members = new ArrayList<Member>();

    // getters, setters, hashcode, equals
}

Страница JSF:

<h:form>
    <h:selectManyCheckbox id="stat" value="#{memberModif.member.listeStatut}">
        <f:converter converterId="statutConverter"/>
        <f:selectItems value="#{memberModif.statutsPossibles}" var="statut" itemValue="#{statut}" itemLabel="#{statut.name}" />
    </h:selectManyCheckbox>


    <h:commandLink id="register" action="#{memberModif.modifier()}" value="Modifier">
        <f:param name="cid" value="#{javax.enterprise.context.conversation.id}"/>
    </h:commandLink>
</h:form>

компонент поддержки (я пытался с ConversationScoped после SessionScoped -> та же проблема)

@ConversationScoped
@Named
public class MemberModif implements Serializable {

    private static final long serialVersionUID = -291355942822086126L;

    @Inject
    private Logger log;

    @Inject
    private EntityManager em;

    @Inject Conversation conversation;

    private Member member;

    @SuppressWarnings("unused")
    @PostConstruct
    private void init() {
        if (conversation.isTransient()) {
            conversation.begin();
        }
    }

    public String modifier() {
        em.merge(member);
    }

    public Member getMember() {
        if (member == null) {
            member = em.createQuery("from Member m where m.id=:id",Member.class).setParameter("id", new Long(0)).getSingleResult();
        }
        return member;
    }

    public List<Statut> getStatutsPossibles() {
        return em.createQuery("from Statut", Statut.class).getResultList();
    }
}

И конвертер (сильно вдохновленный швом ObjectConverter):

@FacesConverter("statutConverter")
public class StatutConverter implements Converter, Serializable {

    final private Map<String, Statut> converterMap = new HashMap<String, Statut>();
    final private Map<Statut, String> reverseConverterMap = new HashMap<Statut, String>();

    @Inject
    private transient Conversation conversation;

    private final transient Logger log = Logger.getLogger(StatutConverter.class);

    private int incrementor = 1;

    @Override
    public Object getAsObject(FacesContext context, UIComponent component, String value) {
        if (this.conversation.isTransient()) {
            log.warn("Conversion attempted without a long running conversation");
        }

        return this.converterMap.get(value);
    }

    @Override
    public String getAsString(FacesContext context, UIComponent component, Object value) {
        if (this.conversation.isTransient()) {
            log.warn("Conversion attempted without a long running conversation");
        }

        if (this.reverseConverterMap.containsKey(value)) {
            return this.reverseConverterMap.get(value);
        } else {
            final String incrementorStringValue = String.valueOf(this.incrementor++);
            this.converterMap.put(incrementorStringValue, (Statut)value);
            this.reverseConverterMap.put( (Statut)value, incrementorStringValue);
            return incrementorStringValue;
        }
    }
}

Обратите внимание, что я разместил этот конвертер здесь, чтобы избежать поиска в сети для реализации шва, но он аналогичен использованию тега <s:objectConverter/> вместо <f:converter converterId="statutConverter"/>

Любая помощь будет принята с благодарностью.

Ответы [ 2 ]

1 голос
/ 06 февраля 2014

Взгляните на это: selectManyCheckbox LazyInitializationException при проверке процесса

Попробуйте: <f:attribute name="collectionType" value="java.util.ArrayList" />; на вашем <h:selectManyCheckbox>

1 голос
/ 08 июля 2012

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

Hibernate.initialize(yourCollection);  
...