在每个页面刷新时调用Session Scoped Managed Bean构造函数

时间:2012-05-27 19:22:04

标签: jsf

我正在使用会话范围的托管bean来处理Java EE应用程序中的登录。在对用户进行身份验证后,用户对象将保存在此会话Bean中。但是,刷新页面后,会话bean值就消失了。

我正在调试代码,结果是在页面刷新时再次调用会话作用域托管bean的构造函数,因此使用新用户初始化用户对象。我想这不是一个正常的行为,因为它应该保存在会话上不应该吗?

我发布了登录托管bean的一些部分,包括参数和登录方法。基本上,enteredEmail和enteredPassword代表登录表单上输入的数据。如果身份验证成功,则loggedIn布尔值将变为true,并且登录的用户对象将存储在checkedUser变量中。

import javax.faces.bean.ManagedBean;
import javax.faces.bean.SessionScoped;

@ManagedBean
@SessionScoped
public class LoginController  implements Serializable {

@EJB
private LoginSessionBean loginSessionBean;
@EJB
private LecturerFacade lecturerFacade;

private Lecturer checkedUser;
private String enteredEmail;
private String enteredPassword;
private boolean loggedIn;





/** Creates a new instance of loginController */
public LoginController() {
   loggedIn = false;
   checkedUser = new Lecturer();
}
public String login(){
    RequestContext context = RequestContext.getCurrentInstance();
    FacesMessage msg = null;

    this.setCheckedUser(lecturerFacade.findLecturerByEmail(enteredEmail));

    if(loginSessionBean.checkPassword(checkedUser, enteredPassword))
    {
        loggedIn = true;
        msg = new FacesMessage(FacesMessage.SEVERITY_INFO, "Welcome", checkedUser.getFirstName()+ " " + checkedUser.getLastName());



        FacesContext.getCurrentInstance().addMessage(null, msg);
        context.addCallbackParam("loggedIn", loggedIn);



    }
    return "Index";

我也发布了上面托管bean使用的两个EJB。讲师使用输入的电子邮件检索用户对象,而loginSessionBean则检查密码。

@Stateless
public class LecturerFacade extends AbstractFacade<Lecturer> {
@PersistenceContext(unitName = "EffectinetWebPU")
private EntityManager em;

Logger logger = Logger.getLogger("MyLog");
FileHandler fh;


protected EntityManager getEntityManager() {
    return em;
}

public LecturerFacade() {
    super(Lecturer.class);
}

public Lecturer findLecturerByEmail(String email) {
    try {
        return (Lecturer) this.getEntityManager().createQuery("SELECT l FROM Lecturer l WHERE l.email = :email").setParameter("email", email).getSingleResult();
    } catch (NoResultException e) {
        System.err.println("Caught NOResultException: "+  e.getMessage());
        return null;
    } catch (NonUniqueResultException e) {
        System.err.println("Caught NonUniqueResultException: "+  e.getMessage());
        return null;
    } catch (IllegalStateException e) {
        System.err.println("Caught IllegalStateException: "+  e.getMessage());
        return null;
    }
}

_

@Stateless

public class LoginSessionBean {

// Add business logic below. (Right-click in editor and choose
// "Insert Code > Add Business Method")
@PersistenceContext(unitName = "EffectinetWebPU")
private EntityManager em;

protected EntityManager getEntityManager() {
    return em;
}

public void setEntityManager(EntityManager em) {
    this.em = em;
}



public boolean checkPassword(Lecturer user, final String enteredPassword) {
    if (user.getPassword().equals(enteredPassword)) {
        return true;
    } else {
        return false;
    }
}

}

如果有人对出了什么问题有任何建议,请告诉我

我使用glassfish 3.1作为应用程序服务器,使用Primefaces作为JSF库。另外,我已检查并从右侧包中导入sessionScoped注释,而不是从javax.enterprise ...

2 个答案:

答案 0 :(得分:6)

你的问题就在这里:

<p:menuitem value="Logout" ... onclick="#{loginController.logout()}"/>

onclick属性应代表 JavaScript 处理函数,当最终用户点击元素时,该函数将在webbrowser中执行。像

这样的东西
onclick="alert('You have clicked this element!')"

onclick属性也接受ValueExpression,因此您甚至可以让JSF / EL相应地自动生成其值:

onclick="#{bean.onclickFunction}"

public String getOnclickFunction() {
    return "alert('You have clicked this element!')";
}

因此在呈现页面时评估所有EL。在您的特定情况下,每次评估EL时都会调用logout()方法,因此每次呈现页面时都会使会话无效!

您需要将其绑定到一个属性MethodExpression,如<h:commandLink action><h:commandButton action>,在此特定情况下为<p:menuitem action>

<p:menuitem value="Logout" ... action="#{loginController.logout()}"/>

这可以通过理解基本的HTML和JavaScript概念来理解,并记住JSF最终生成HTML / CSS / JS。在webbrowser中打开JSF页面,右键单击查看源以实现它。

答案 1 :(得分:0)

我今天成功解决了这个问题。这是问题,虽然我无法解释原因:

我使用Primefaces 3.2作为JSF库,因此这是索引页面的主菜单。

<h:form>
        <p:menubar >
            <p:menuitem id="registerLink" value="Register" rendered="#{!loginController.loggedIn}" onclick="registerDialog.show()" /> 

            <p:menuitem id="loginLink" value="Login" rendered="#{!loginController.loggedIn}" onclick="loginDialog.show()" />

            <p:submenu label="Units" rendered="true"> 
                <p:menuitem id="addNew" value="Add New" onclick="createUnitDialog.show()" />

                <p:menuitem id="myUnits" value="My Units" onclick="" />
            </p:submenu>        

            <p:menuitem id="results" value="Results/Statistics" rendered="#{loginController.loggedIn}" onclick=""/>

            <p:menuitem id="profile" value="My Profile" rendered="#{loginController.loggedIn}" onclick=""/>

            <p:menuitem id="logout" value="Logout" rendered="#{loginController.loggedIn}" onclick="#{loginController.logout()}"/>

        </p:menubar>
    </h:form>

在为整个代码设置断点之后,我发现每次刷新页面都会调用logout()方法,该方法应该销毁托管bean。我不知道为什么会发生这种情况,因为在点击退出menuitem时应该调用它。

但是,在使用onclick="#{loginController.logout()}更改action="#{loginController.logout()}后,问题就解决了。

我查看了Primefaces的文档,但没有解释这种行为