为什么GlassFish 5.1.0中的JSF将我的@ViewScoped CDI bean的ID属性设置为null?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了为什么GlassFish 5.1.0中的JSF将我的@ViewScoped CDI bean的ID属性设置为null?相关的知识,希望对你有一定的参考价值。

我正在从GlassFish 4.1.2升级到GlassFish 5.1.0,并遇到一个问题,即JSF生命周期的更新模型阶段将我的@ViewScoped CDI支持bean的ID属性的值设置为null。以下是我的简化视图。

<ui:composition xmlns="http://www.w3.org/1999/xhtml"
xmlns:ui="http://xmlns.jcp.org/jsf/facelets"
xmlns:f="http://java.sun.com/jsf/core"
xmlns:h="http://xmlns.jcp.org/jsf/html"
xmlns:p="http://primefaces.org/ui"
template="/WEB-INF/templates/modena.xhtml">
<ui:define name="title">Location Editor2</ui:define>
<ui:define name="content">
    <f:metadata>
        <f:viewParam name="id" value="#{locationEditor2.location.id}" />
        <f:viewAction action="#{locationEditor2.setLocationById}" />
    </f:metadata>
    <h:form id="location-editor-form">
        <p:commandButton value="Update" styleClass="RaisedButton"
            action="#{locationEditor2.updateLocation}" update="@form" />
    </h:form>
</ui:define>
</ui:composition>

以下是我简化的CDI支持bean。

import java.io.Serializable;

import javax.ejb.EJB;
import javax.ejb.EJBException;
import javax.faces.application.FacesMessage;
import javax.faces.context.FacesContext;
import javax.faces.view.ViewScoped;
import javax.inject.Named;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.elliottlogic.elis.ejb.location.Location;
import com.elliottlogic.elis.ejb.location.LocationManager;
import com.elliottlogic.elis.ejb.security.AuthorizationException;
import com.elliottlogic.elis.wa.core.ExceptionTools;

/**
 * The <code>LocationEditor2</code> class TODO Complete Type Description
 *
 * @author jre
 * @version $Id: $
 *
 */
@Named
@ViewScoped
public class LocationEditor2 implements Serializable {

final static Logger logger = LoggerFactory.getLogger(LocationEditor2.class.getName());

static final long serialVersionUID = 1L;

private Location location = new Location();

@EJB
private LocationManager locationManager;

/**
 * Configures model using persisted values.
 * 
 * @throws AuthorizationException
 */
public void setLocationById() throws AuthorizationException {

    try {

        location = locationManager.readLocationByID(location.getId());

    } catch (EJBException e) {

        FacesContext.getCurrentInstance().addMessage(null,
                new FacesMessage(FacesMessage.SEVERITY_ERROR, "Location not found.", "Location not found."));

        ExceptionTools.unwrapEJBException(FacesContext.getCurrentInstance(), e);

    }

}

/**
 * Saves the changes to the location.
 * 
 * @return
 */
public String updateLocation() {

    logger.debug("Location ID: {}", location.getId());

    return null;

}

/**
 * @return the location
 */
public Location getLocation() {
    return location;
}

/**
 * @param location
 *            the location to set
 */
public void setLocation(Location location) {
    this.location = location;
}

}

以下演示了此问题:

  1. 位于LocationEditor2.xhtml的点浏览器?id = 1
  2. 选择[更新]按钮。
  3. updateLocation方法记录“位置ID:1”
  4. 选择[更新]按钮
  5. updateLocation方法记录“Location ID:null”

我验证了视图没有被重新创建,并且在Update Model JSF阶段,某些东西正在调用值为null的setId(Long id)方法。

为什么JSF更新模型阶段在第二次和附加回发期间将我的实体的ID设置为null?

答案

问题的根本原因是,在JSF生命周期的更新模型阶段,某些东西正在我的实体上调用setId(null)方法。我在setId()方法中使用了StackTraceElement []数组来查看调用方法,并确定UIViewParameter类的updateModel方法是罪魁祸首。

经过一番谷歌搜索后,我发现OmniFaces viewParam Showcase提供了“无状态模式,以避免在回发上进行不必要的转换,验证和模型更新”。在更改我的代码以使用OmniFaces viewParam组件后,JSF更新模型基础结构在回发期间停止在我的实体上调用setId()方法,并且一切都像在GlassFish 4.1.2中那样工作。

解决方案是从标准的f:viewParam移动到OmniFaces o:viewParam。

以上是关于为什么GlassFish 5.1.0中的JSF将我的@ViewScoped CDI bean的ID属性设置为null?的主要内容,如果未能解决你的问题,请参考以下文章

为啥我的 Glassfish3.1.2.2/MyFaces2.1.9/JSF 管理的性能优于 TomEE1.5+/CDI 管理的性能?

IDEA 2020.2 部署JSF项目

JSF2 + Eclipse + Glassfish 奇怪的输出问题

为啥选择浏览器刷新按钮不会将我的 JSF 表单中的已编辑字段替换为来自服务器的值? [复制]

在 Glassfish 3.0 安全领域中使用 bcrypt

GlassFish 5 上带有 JSF 2.3 Web 应用程序的上下文根