亚洲在线久爱草,狠狠天天香蕉网,天天搞日日干久草,伊人亚洲日本欧美

為了賬號安全,請及時綁定郵箱和手機立即綁定
已解決430363個問題,去搜搜看,總會有你想問的

如何在 Undertow 的非阻塞處理程序中執行阻塞代碼?

如何在 Undertow 的非阻塞處理程序中執行阻塞代碼?

慕俠2389804 2023-03-09 15:27:10
正如在一個單獨的問題中所描述的,當使用 Undertow 時,所有的處理都應該在一個專用的 Worker 線程池中完成,它看起來像這樣:public class Start {  public static void main(String[] args) {    Undertow server = Undertow.builder()        .addListener(8080, "localhost")        .setHandler(new HttpHandler() {          public void handleRequest(HttpServerExchange exchange)              throws Exception {            if (exchange.isInIoThread()) {              exchange.dispatch(this);              return;            }            exchange.getResponseHeaders()                    .put(Headers.CONTENT_TYPE, "text/plain");            exchange.getResponseSender()                    .send("Hello World");          }        })        .build();    server.start();  }}我知道這BlockingHandler可以用于明確告訴 Undertow 將請求安排在專用線程池上以阻止請求。HttpHandler我們可以通過將 包裝在 的實例中來調整上面的示例BlockingHandler,如下所示:        .setHandler(new BlockingHandler(new HttpHandler() {這適用于我們知道總是阻塞的呼叫。但是,如果某些代碼大部分時間是非阻塞的,但有時需要阻塞調用,如何將阻塞調用變成非阻塞調用?例如,如果請求的值存在于緩存中,則以下代碼不會阻塞(它只是從 some 中獲取Map<>),但如果不存在,則必須從數據庫中獲取。public class Start {  public static void main(String[] args) {    Undertow server = Undertow.builder()        .addListener(8080, "localhost")        .setHandler(new HttpHandler() {          public void handleRequest(HttpServerExchange exchange)              throws Exception {            if (exchange.isInIoThread()) {              exchange.dispatch(this);              return;            }            if (valueIsPresentInCache(exchange)) {              return valueFromCache;  // non-blocking            } else {              return fetchValueFromDatabase(); // blocking!!!            }          }        })        .build();    server.start();  }}根據文檔,有一個方法HttpServerExchange.startBlocking(),但根據 JavaDoc,除非你真的需要使用輸入流,否則這個調用仍然是一個阻塞的。調用此方法將交換置于阻塞模式,并創建一個 BlockingHttpExchange 對象來存儲流。當交換器處于阻塞模式時,輸入流方法變得可用,除此之外,阻塞模式和非阻塞模式之間目前沒有主要區別如何將這個阻塞調用變成非阻塞調用?
查看完整描述

1 回答

?
開心每一天1111

TA貢獻1836條經驗 獲得超13個贊

正確的方法是在 IO 線程中實際執行邏輯,如果它是非阻塞的。否則,將請求委托給專用線程,如下所示:


public class Example {


  public static void main(String[] args) {

    Undertow server = Undertow.builder()

        .addListener(8080, "localhost")

        .setHandler(new HttpHandler() {

          public void handleRequest(HttpServerExchange exchange)

              throws Exception {


            if (valueIsPresentInCache(exchange)) {

              getValueFromCache();  // non-blocking, can be done from IO thread           

            } else {


              if (exchange.isInIoThread()) {

                exchange.dispatch(this);

                // we return immediately, otherwise this request will be

                // handled both in IO thread and a Worker thread, throwing

                // an exception

                return;

              }


              fetchValueFromDatabase(); // blocking!!!


            }

          }

        })

        .build();

    server.start();

  }

}


查看完整回答
反對 回復 2023-03-09
  • 1 回答
  • 0 關注
  • 135 瀏覽
慕課專欄
更多

添加回答

舉報

0/150
提交
取消
微信客服

購課補貼
聯系客服咨詢優惠詳情

幫助反饋 APP下載

慕課網APP
您的移動學習伙伴

公眾號

掃描二維碼
關注慕課網微信公眾號