作用域类型 javax.enterprise.context.RequestScoped 在从线程调用 Bean 时没有活动上下文

2022-09-03 13:29:52

在使用 Weld-SE 2.1.2.Final 获取 Bean 并从线程调用它时,我遇到了以下异常:

线程“main” org.jboss.weld.context.ContextNotActiveException 中的异常:WELD-001303:作用域类型 javax.enterprise.context.RequestScoped 没有活动上下文

我的豆子用@RequestScooped注释。如果我@ApplicationScoped注释,那么它工作正常,但我需要保持@RequestScooped。

这是一个复制器:

public static void main(String[] args) throws Exception {
    Weld weld = new Weld();
    WeldContainer container = weld.initialize();
    final MyPojo pojo = container.instance().select(MyPojo.class).get();

    Thread t = new Thread() {
        public void run() {
            System.out.println(pojo.ping());   // This call fails
        }
    };
    t.start();
    t.join();
    System.out.println(pojo.ping()); // This call succeed
    weld.shutdown();

}

@RequestScoped
public class MyPojo {
 public String ping() {
    return "pong";
 }
}

您是否遇到过此行为?有什么想法可以使这个工作吗?


答案 1

在这种情况下,Weld 使用的是与线程关联的未绑定 RequestContext(RequestContext)。您需要在您正在创建的线程中手动初始化新的 RequestContext,这对我有用:

public static void main(String[] args) throws Exception {
    Weld weld = new Weld();
    final WeldContainer container = weld.initialize();
    RequestContext requestContext= container.instance().select(RequestContext.class, UnboundLiteral.INSTANCE).get();
    requestContext.activate();

    final MyPojo pojo = container.instance().select(MyPojo.class).get();

    Thread t = new Thread() {
        public void run() {
            RequestContext requestContext= container.instance().select(RequestContext.class, UnboundLiteral.INSTANCE).get();
            requestContext.activate();
            System.out.println("1" + pojo.ping()); 
        }
    };
    t.start();
    t.join();
    System.out.println("2" + pojo.ping());
    weld.shutdown();

}

答案 2