我试图通过在Runnable类的run()方法中调用FacesContext.getCurrentInstance()来获取FacesContext,但它返回null.
public class Task implements Runnable { @Override public void run() { FacesContext context = FacesContext.getCurrentInstance(); // null! // ... } }
这是怎么造成的,我该如何解决?
解决方法
FacesContext
作为
ThreadLocal
变量存储在负责调用FacesServlet的HTTP请求的线程中,FacesServlet负责创建FacesContext.该线程通常仅通过JSF托管bean方法. FacesContext在该线程生成的其他线程中不可用.
实际上你应该也不需要在其他线程中使用它.此外,当您的线程启动并独立运行时,底层HTTP请求将立即继续处理HTTP响应然后消失.无论如何,您将无法使用HTTP响应执行某些操作.
您需要以不同方式解决您的问题.问问自己:你需要什么?获取一些信息?只需在构建过程中将该信息传递给Runnable即可.
以下示例假定您要访问线程中的某个会话作用域对象.
public class Task implements Runnable { private Work work; public Task(Work work) { this.work = work; } @Override public void run() { // Just use work. } }
Work work = (Work) FacesContext.getCurrentInstance().getExternalContext().getSessionMap().get("work"); Task task = new Task(work); // ...
但是,如果您最终需要通知客户,例如线程的工作已经完成,那么你应该寻找一个不同于添加面部消息左右.答案是使用“推”.这可以通过SSE或websockets实现.在这个相关问题中可以找到具体的websockets示例:Real time updates from database using JSF/Java EE.如果您碰巧使用PrimeFaces,请查看<p:push>
.如果您碰巧使用OmniFaces,请查看<o:socket>
.
与具体问题无关,手动创建Runnables并在Java EE Web应用程序中手动生成线程令人震惊.前往以下Q& A,了解所有警告以及如何实际完成:
> Spawning threads in a JSF managed bean for scheduled tasks using a timer
> Is it safe to start a new thread in a JSF managed bean?