【问题标题】:How do I get a SessionScoped CDI bean from inside a Filter?如何从过滤器中获取 SessionScoped CDI bean?
【发布时间】:2011-12-10 13:07:59
【问题描述】:

此问题与writing a session timeout handler 上的上一个问题有关。

该线程中的答案涉及从 servlet 访问各种会话范围的托管 bean。建议(如 here 所示)是在过滤器中执行此操作:

HttpSession session = request.getSession(false);
User user = (session != null) ? (User) session.getAttribute("user") : null;

大概这会获取 User 类的会话 bean。问题是这不起作用。

问题在于 bean 存在于会话属性中,但它们是由 Weld 工具包装的。我写了 doFilter() 方法如下:

public void doFilter(ServletRequest request, ServletResponse response,
        FilterChain chain)
        throws IOException, ServletException {
    HttpServletRequest req = (HttpServletRequest) request;
    HttpServletResponse res = (HttpServletResponse) response;

    String sp = req.getServletPath();
    System.out.println("------------------------");
    System.out.println("doFilter(): " + sp);

    if (!sp.startsWith("/javax")) {  // eliminates many requests
        HttpSession session = req.getSession();
        Enumeration<String> en = session.getAttributeNames();
        int count = 0;            
        while (en.hasMoreElements()) {
            String e = en.nextElement();
            System.out.println("Attribute " + ++count + ": " + e);
        }
    }
    chain.doFilter(request, response);
}

当这转储会话属性时,我通常会得到这样的结果:

INFO: ------------------------
INFO: doFilter(): /Display.xhtml
INFO: Attribute 1: org.jboss.weld.context.http.HttpSessionContext#org.jboss.weld.bean-WEB-INF/lib/myfaces-extcdi-bundle-jsf20-1.0.1-ManagedBean-class org.apache.myfaces.extensions.cdi.jsf.impl.scope.conversation.EditableWindowContextManagerProxy
INFO: Attribute 2: org.jboss.weld.context.http.HttpSessionContext#org.jboss.weld.bean-MyApp5-ManagedBean-class com.app.Login
INFO: Attribute 3: org.jboss.weld.context.conversation.ConversationIdGenerator
INFO: Attribute 4: com.sun.faces.renderkit.ServerSideStateHelper.LogicalViewMap
INFO: Attribute 5: org.jboss.weld.context.ConversationContext.conversations
INFO: Attribute 6: facelets.ui.DebugOutput
INFO: Attribute 7: javax.faces.request.charset
INFO: Attribute 8: org.apache.myfaces.extensions.cdi.core.api.scope.conversation.WindowContext:EXISTING_WINDOW_ID_LIST

属性 #2 似乎代表了我想要的 bean。不用说,调用 session.getAttribute("login") 不起作用。

谁能说出如何访问底层托管 bean?我更愿意以一种与 Weld 无关的方式来做这件事,但这可能是不可能的。

【问题讨论】:

    标签: jsf-2 servlet-filters cdi


    【解决方案1】:

    此方法仅适用于会话范围的 JSF @ManagedBean,不适用于 CDI @Named bean。

    你需要@Inject它作为过滤器的一个属性。

    @Inject
    private User user;
    

    【讨论】:

    • BalusC 你睡过吗?我刚刚尝试用 @Named 注释我的过滤器,并为我的 Login bean 做一个 @Inject。有效!我不知道 CDI 服务会在 Filter 中工作。我想这意味着 Filter 对象在每个请求周期都被实例化,而不仅仅是在 Servlet 的生命周期内实例化一次。
    • 现在只有 21:02 :) 如果您使用 @WebFilter 注释过滤器,它应该可以工作。我宁愿不让它@Named(同样,@Inject 也应该在@WebServlet servlet 中工作)。 CDI 更像是一个代理。对象是在线程本地注入的,要注入的实例不需要具有相同或更广泛的范围。
    • @BalusC: 当user 成为过滤器的成员变量时,这个解决方案线程安全吗?
    • @Toru: 除非你不省略@Inject
    【解决方案2】:
    import org.jboss.weld.context.SerializableContextualInstanceImpl;
    
        HttpSession httpSession = (HttpSession) facesContext.getExternalContext().getSession(false);
        Enumeration<String> attribs = httpSession.getAttributeNames();
        String attrib = null;
        while (attribs.hasMoreElements()) {         
            attrib = attribs.nextElement();
            Object obj = httpSession.getAttribute(attrib);
            if(obj instanceof SerializableContextualInstanceImpl){
                SerializableContextualInstanceImpl impl = (SerializableContextualInstanceImpl)obj;
                //here: MyObject myObj= (MyObject)impl.getInstance();
            }           
        }
    

    【讨论】:

      猜你喜欢
      • 2013-09-05
      • 2020-03-16
      • 2019-08-04
      • 2011-07-11
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2012-04-23
      相关资源
      最近更新 更多