Files
cpp-httplib/docs-src/pages/ja/cookbook/s16-disconnect.md
2026-04-10 18:47:42 -04:00

2.3 KiB

title, order, status
title order status
S16. クライアントが切断したか検出する 35 draft

長時間のレスポンスを返している最中に、クライアントが接続を切ってしまうことがあります。無駄に処理を続けても意味がないので、適宜チェックして中断できるようにしておきましょう。cpp-httplibではreq.is_connection_closed()で確認できます。

基本の使い方

svr.Get("/long-task", [](const httplib::Request &req, httplib::Response &res) {
  for (int i = 0; i < 1000; ++i) {
    if (req.is_connection_closed()) {
      std::cout << "client disconnected" << std::endl;
      return;
    }

    do_heavy_work(i);
  }

  res.set_content("done", "text/plain");
});

is_connection_closed()std::function<bool()>なので、()を付けて呼び出します。切断されていればtrueを返します。

ストリーミングレスポンスと組み合わせる

set_chunked_content_provider()でストリーミング配信しているときも、同じ方法で切断を検出できます。ラムダにキャプチャしておくと便利です。

svr.Get("/events", [](const httplib::Request &req, httplib::Response &res) {
  res.set_chunked_content_provider(
    "text/event-stream",
    [&req](size_t offset, httplib::DataSink &sink) {
      if (req.is_connection_closed()) {
        sink.done();
        return true;
      }

      auto event = generate_next_event();
      sink.write(event.data(), event.size());
      return true;
    });
});

切断を検出したらsink.done()を呼んで、プロバイダの呼び出しを止めます。

どのくらいの頻度でチェックすべきか

毎ループで呼んでも軽い処理ですが、あまりに細かい単位で呼ぶと意味が薄れます。「1チャンクを生成し終えたタイミング」や「データベースの1クエリが終わったタイミング」など、中断しても安全な境目で確認するのが現実的です。

Warning: is_connection_closed()の確認は即座に正確な値を返すとは限りません。TCPの特性上、送信が止まって初めて切断に気付くことも多いです。完璧なリアルタイム検出を期待せず、「そのうち気付ければいい」くらいの気持ちで使いましょう。