0

我正在使用 maven、jsf 2.0、java ee 6、jpa 和 glassfish。我有一个初始网页,询问您是否要登录或注册。如果他们说他们想登录,他们会被转发到 facelets/primefaces 3,它会显示一个带有用户名和密码文本框的登录页面。对于用户名文本字段,其值为

value="#{authenticator.username}"

我得到一个

“目标无法访问,标识符‘验证器’解析为空”

我正在使用 beans.xml。我遵循 CDI bean 的命名约定。我在 Authenticator 构造函数中插入了一条日志消息,它似乎永远不会被调用,因为我没有收到消息。

另一个奇怪的是,一旦在一个蓝月亮它工作,我得到如下的东西:

07/08/2012 00:27:56.140 信息 ...Authenticator - 在 Authenticator 构造函数中

2012 年 7 月 8 日 00:27:56.171 信息 ...Authenticator - 获取 Authenticator 用户名

2012 年 7 月 8 日 00:27:56.171 信息 ...Authenticator - 在 Authenticator 构造函数中

2012 年 7 月 8 日 00:27:56.171 信息 ...验证器 - 获取验证器密码

2012 年 7 月 8 日 00:27:56.171 信息 ...Authenticator - 在 Authenticator 构造函数中

2012 年 7 月 8 日 00:27:56.171 信息 ...Authenticator - 获取 Authenticator 用户名

2012 年 7 月 8 日 00:27:56.171 信息 ...Authenticator - 在 Authenticator 构造函数中

2012 年 7 月 8 日 00:27:56.171 信息 ...验证器 - 获取验证器密码

2012 年 7 月 8 日 00:28:05.843 信息 ...Authenticator - 在 Authenticator 构造函数中

2012 年 7 月 8 日 00:28:05.843 信息 ...Authenticator - 在 Authenticator 构造函数中

2012 年 7 月 8 日 00:28:05.843 信息 ...Authenticator - 在 Authenticator 构造函数中

07/08/2012 00:28:05.843 INFO ...Authenticator - 获取 Authenticator 用户名

2012 年 7 月 8 日 00:28:05.843 信息 ...Authenticator - 在 Authenticator 构造函数中

2012 年 7 月 8 日 00:28:05.906 信息 ...Authenticator - 在 Authenticator 构造函数中

07/08/2012 00:28:05.906 INFO ...验证器 - 获取验证器密码

2012 年 7 月 8 日 00:28:26.000 信息 ...Authenticator - 在 Authenticator 构造函数中

2012 年 7 月 8 日 00:28:26.000 信息 ...Authenticator - 在 Authenticator 构造函数中

2012 年 7 月 8 日 00:28:26.000 信息 ...Authenticator - 在 Authenticator 构造函数中

07/08/2012 00:28:26.031 INFO ...Authenticator - 在 Authenticator 构造函数中 07/08/2012 00:28:26.031 INFO ...Authenticator - 获取 Authenticator 密码

当它工作时,我不明白为什么它会这么多次调用构造函数。

这是托管(cdi)bean:

package com.mlb.mybills.view.user;

import com.mlb.mybills.i18n.Messages;
import java.util.Locale;
import javax.faces.application.FacesMessage;
import javax.faces.bean.ViewScoped;
import javax.faces.context.ExternalContext;
import javax.faces.context.FacesContext;
import javax.inject.Named;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

@Named("authenticator")
@ViewScoped
public class Authenticator
{
    private static final Logger log = LoggerFactory.getLogger(Authenticator.class);

    private String username;
    private char[] password;

    public Authenticator()
    {
      log.info("in Authenticator constructor");
    }

    public String getUsername()
    {
      log.info("getting Authenticator username");
        return username;
    }

    public void setUsername(String username)
    {
      log.info("getting Authenticator username");
        this.username = username;
    }

    public char[] getPassword()
    {
      log.info("getting Authenticator password");
        return password;
    }

    public void setPassword(char[] password)
    {
        log.info("setting Authenticator password");
      this.password = password;
    }

public void setPassword(String password)
    {
        log.info("setting Authenticator password");
      this.password = password.toCharArray();
    }


   public String authenticate()
    {
      log.info("in Authenticator.authenticate");
        String result = null;
        ExternalContext externalContext = FacesContext.getCurrentInstance().getExternalContext();
        HttpServletRequest request = (HttpServletRequest) externalContext.getRequest();
        try
        {
            request.login(username, new String(password));
            //result = "/private/group.xhtml?faces-redirect=true";
         result = "/group.xhtml?faces-redirect=true";
        }
        catch (ServletException ex)
        {
            log.error("Failed to authenticate user.", ex);
            Locale locale = FacesContext.getCurrentInstance().getViewRoot().getLocale();
         FacesMessage facesMessage = new FacesMessage(FacesMessage.SEVERITY_ERROR, Messages.getString(
                    "Login.InvalidIdPasswordMessage", locale), null);
            FacesContext.getCurrentInstance().addMessage(null, facesMessage);
        }
      log.info("result=" + result);
        return result;
    }

    public String logout()
    {
        ExternalContext externalContext = FacesContext.getCurrentInstance().getExternalContext();
        HttpServletRequest request = (HttpServletRequest) externalContext.getRequest();
        try
        {
            request.logout();
        }
        catch (ServletException servletEx)
        {
            log.warn("Failed to logout the user", servletEx);
        }
        return "/Login.xhtml?faces-redirect=true";
    }
}
4

3 回答 3

1

您遇到了一些问题,因为您混合了 CDI (@Named) 和 JSF (@ViewScoped) 注释。由于这场比赛,它可能被多次调用。您可以查看 Seam Faces、MyFaces CODI 或为 CDI 推出您自己的 ViewScoped 版本,或者放弃 CDI 并直接使用 JSF。

于 2012-07-09T16:57:08.210 回答
1

您正在混合 bean 管理 API。来自@NamedCDI,但@ViewScoped来自 JSF。您需要使用其中一个,而不是在单个 backing bean 上同时使用。现在 bean 基本上由 CDI 管理,没有任何有效范围,因此每次#{authenticator}解析 EL 表达式时都会构造 bean。

要解决此问题,如果您打算坚持使用 CDI,请使用@ConversationScoped代替@ViewScoped

@Named
@ConversationScoped

或者,如果您打算坚持使用 JSF,请使用@ManagedBean代替@Named

@ManagedBean
@ViewScoped

CDI 方法的唯一缺点是您必须将@InjectaConversation作为属性,然后自己开始和结束它。

于 2012-07-09T15:14:25.520 回答
0

尽管我不明白,但我为我找到了解决方案。我在以下链接中找到了它:一个链接!我不仅在 faces-config.xml 中添加了以下条目,而且还添加了 @RequestScoped 和 @Named

<managed-bean>
   <managed-bean-name>authenticator</managed-bean-name>
   <managed-bean-class>com.mlb.mybills.view.user.Authenticator</managed-bean-class>
   <managed-bean-scope>session</managed-bean-scope>
</managed-bean>

有了这个,我不再得到目标无法到达的错误。

显然,我还纠正了 BalusC 指出的上述错误。谢谢大家。

于 2012-07-22T23:51:24.520 回答