我正在使用会话范围的托管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 ...
答案 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的文档,但没有解释这种行为