【问题标题】:Init method gets called twice in @Viewscoped beanInit 方法在 @Viewscoped bean 中被调用两次
【发布时间】:2012-12-26 04:56:03
【问题描述】:

我正在使用 Primefaces 的日程安排组件。我正在用数据库中的值填充它,当用户从 selectonemenu 中选择某个 ajax 事件时,会触发一个 ajax 事件(我试图只放置相关代码,如果缺少某个东西请提醒我):

xhtml:

 <h:outputText value="Scope :" />
 <h:selectOneMenu id="scope" value="#{scheduleView.scope}">                    
   <f:selectItems value="#{lookup.scopeCombo}"/>
   <p:ajax process="scope" update="schedule, scheduleForm, scheduleFormPG" listener="#{scheduleView.changeScopeType()}"/>
 </h:selectOneMenu> 

<p:schedule id="schedule" value="#{scheduleView.model}" editable="true"/>

支持 bean:

@ManagedBean
@ViewScoped
public class ScheduleView implements Serializable { 

@PostConstruct
    public void init() {
        System.out.println("Init ");
        scopeChange();                
    } 

    public void scopeChange(String scope){
        System.out.println("scopeChange ");
        model.clear();
        events = (List<Event>) commonServis.bringEverythingByCriteria(Event.class, "scope" , scope);   
        for(int i= 0; i<events.size();i++){
        model.addEvent(new DefaultScheduleEvent(events.get(i).getAd(), events.get(i).getStartDate(),events.get(i).getEndDate()));
    }

    public void changeScopeType() {
    System.out.println("changeScopeType ");
        scopeChange (scope);
}

以上代码的输出为:

Init
scopeChange

当用户更改选择菜单中的值时:

changeScopeType
Init
scopeChange

它应该只进入init 方法一次。但是在changeScopeType 函数被触发后,它会进入init 方法并用不相关的数据填充计划。我认为它可能与@Postconstruct 注释有关,但我找不到任何相关解释。任何人都可以理解原因并提供解决方案吗?

这是完整的页面:

<?xml version='1.0' encoding='UTF-8' ?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<ui:composition xmlns="http://www.w3.org/1999/xhtml"
                xmlns:h="http://java.sun.com/jsf/html"
                xmlns:f="http://java.sun.com/jsf/core"
                xmlns:ui="http://java.sun.com/jsf/facelets"
                xmlns:p="http://prime.primefaces.org/ui"
                template="templates/layout.xhtml"
                xmlns:c="http://java.sun.com/jsp/jstl/core">
    <ui:define name="title">#{labels.schedule}</ui:define>
    <ui:define name="content">
        <h:form id="scheduleForm">
            <h:panelGrid id="scheduleFormPG">
                <p:growl id="msgs" />
                <p:dialog modal="true" widgetVar="statusDialog" header="Status" draggable="false">
                    <h:graphicImage value="resources/images/ajax-loader.gif" />
                </p:dialog>
                <p:dialog showEffect="explode" hideEffect="explode" resizable="false"
                      header="warning" widgetVar="confirmationErase" appendToBody="true" modal="true">
                    <h:outputText value="Are you sure?"/>
                    <br/>
                    <p:commandButton  value="Yes" actionListener="#{scheduleView.deleteEvent(AE)}"
                                     update="msgs, scheduleFormPG, schedule, wrapperPanel"
                                     onstart="statusDialog.show(),confirmationErase.hide()"
                                     oncomplete="statusDialog.hide(), eventDialog.hide()" process="@parent, scope" />
                    <p:commandButton value="No" onclick="confirmationErase.hide()" type="button" />
                </p:dialog>                
                <h:outputText value="scope :" />
                <h:selectOneMenu id="scope" value="#{scheduleView.scope}">                    
                    <f:selectItems value="#{lookup.scopeTypeCombo}"/>
                    <p:ajax process="scope" update="schedule, scheduleForm, scheduleFormPG" listener="#{scheduleView.changeScopeType()}"/>
                </h:selectOneMenu>                
             </h:panelGrid>            
            <p:schedule onDateSelectUpdate="wrapperPanel" 
                        onEventSelectUpdate="wrapperPanel" onEventSelectComplete="eventDialog.show()" eventSelectListener="#{scheduleView.onEventSelect}" 
                        onDateSelectComplete="eventDialog.show();"  dateSelectListener="#{scheduleView.onDateSelect}" id="schedule" value="#{scheduleView.model}" editable="true"/>
            <p:dialog id="dialog111" widgetVar="eventDialog" header="Event Information" showEffect="clip" hideEffect="clip">
                <p:panel id="wrapperPanel">                    
                    <h:panelGrid id="eventDetails" columns="2">   
                        <h:outputLabel for="eventName" value="Event Name *: " />
                        <p:inputText id="eventName" value="#{scheduleView.event.title}" required="true"/>
                        <h:outputLabel value="Start Date:" />
                        <p:calendar id="sKalender" value="#{scheduleView.event.startDate}">
                            <f:convertDateTime pattern="dd/MM/yyyy" />  
                        </p:calendar>
                        <h:outputLabel value="End Date:" />
                        <p:calendar id="eKalender" value="#{scheduleView.event.endDate}">
                            <f:convertDateTime pattern="dd/MM/yyyy" />  
                        </p:calendar>
                        <h:outputLabel value="All Day:" />
                        <h:selectBooleanCheckbox id="allDay" value="#{scheduleView.event.allDay}" />
                        <h:outputLabel value="scope: " />
                        <h:selectOneMenu id="scopeChoice" value="#{scheduleView.event.scope}">                    
                            <f:selectItems value="#{lookup.scopeTypeCombo}"/>
                         </h:selectOneMenu>                         
                          <p:commandButton onclick="confirmationErase.show()" oncomplete="eventDialog.hide()" update="wrapperPanel, msgs, schedule" type="reset" value="Delete" />
                        <p:commandButton value="Save" actionListener="#{scheduleView.addEvent(AE)}" process="@parent, scope" update="schedule wrapperPanel msgs scheduleForm" oncomplete="eventDialog.hide();"/>
                    </h:panelGrid>
                </p:panel>
            </p:dialog>
           </h:form>
    </ui:define>
</ui:composition>

【问题讨论】:

  • 您是否在标记处理程序(如 JSTL &lt;c:if&gt;)或 JSF 组件的 idbinding 属性中的任何位置引用 #{scheduleView}?另请参阅此相关问题:stackoverflow.com/questions/14215098/…
  • 感谢您的评论。它不在 JSTL 或 id 或绑定属性中。但它的用法如下: value="#{scheduleView.scope}" 或 actionListener="#{scheduleView.deleteEvent(AE)} 或 listener="#{scheduleView.changeKapsamTipi() 或 render="#{scheduleView.takvimTuruYillik} " 或 eventSelectListener="#{scheduleView.onEventSelect}" 在 p:commandButton、p:schedule、p:dataTable 和 h:selectonemenu 组件中。
  • 好吧,那我看不出任何合理的原因。你用的是弹簧吗?我记得 JSF 部分状态保存机制在安装 Spring 时以某种方式中断,导致所有视图范围的 bean 的行为类似于请求范围的。您需要将javax.faces.PARTIAL_STATE_SAVING 上下文参数显式强制为true
  • 是的,我正在使用 Spring。正如你提到的,我尝试关闭部分状态保存,但它不起作用..
  • 尝试添加到您的 bean @PreDestroy public void destory(){ System.out.println("why ow why"); } 并查看它何时被销毁...

标签: jsf jsf-2 primefaces annotations postconstruct


【解决方案1】:

问题出在这行代码上:

<p:ajax process="scope" update="schedule, scheduleForm, scheduleFormPG" listener="#{scheduleView.changeScopeType()}"/>

更新正在刷新整个页面,而应该只更新时间表。当我将其更改为:

update="schedule" 

问题解决了。 感谢大家的努力。

【讨论】:

    猜你喜欢
    • 2014-12-18
    • 1970-01-01
    • 2014-06-21
    • 2013-09-02
    • 2021-11-04
    • 1970-01-01
    • 2013-12-31
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多