необходимо понять объект io_service boost

#c #multithreading #boost-asio

#c #многопоточность #boost-asio

Вопрос:

учитывая этот пример

 #include <ctime>
#include <iostream>
#include <string>
#include <boost/array.hpp>
#include <boost/bind.hpp>
#include <boost/shared_ptr.hpp>
#include <boost/asio.hpp>
using boost::asio::ip::udp;

std::string make_daytime_string()
{
  using namespace std; // For time_t, time and ctime;
  time_t now = time(0);
  return ctime(amp;now);
}

class udp_server
{
public:
  udp_server(boost::asio::io_serviceamp; io_service, unsigned short port)
     : socket_(io_service, udp::endpoint(udp::v4(), port))
  {
    start_receive();
  }

private:
  void start_receive()
  {
    socket_.async_receive_from(
        boost::asio::buffer(recv_buffer_), remote_endpoint_,
        boost::bind(amp;udp_server::handle_receive, this,
          boost::asio::placeholders::error,
          boost::asio::placeholders::bytes_transferred));
  }

  void handle_receive(const boost::system::error_codeamp; error,
      std::size_t /*bytes_transferred*/)
  {
    if (!error || error == boost::asio::error::message_size)
    {
      boost::shared_ptr<std::string> message(
          new std::string(make_daytime_string()));

      socket_.async_send_to(boost::asio::buffer(*message), remote_endpoint_,
          boost::bind(amp;udp_server::handle_send, this, message,
            boost::asio::placeholders::error,
            boost::asio::placeholders::bytes_transferred));

      start_receive();
    }
  }

  void handle_send(boost::shared_ptr<std::string> /*message*/,
      const boost::system::error_codeamp; /*error*/,
      std::size_t /*bytes_transferred*/)
  {
  }

  udp::socket socket_;
  udp::endpoint remote_endpoint_;
  boost::array<char, 1> recv_buffer_;
};

int main()
{
  try
  {
    boost::asio::io_service io_service;
    std::vector<std::shared_ptr<udp_server>> servers;
    for (unsigned short port = 5000; port != 6000;   port)
    {
        servers.push_back(std::shared_ptr<udp_server>(new 
                          udp_server(io_service, port)));
    }
    io_service.run();
  }
  catch (std::exceptionamp; e)
  {
    std::cerr << e.what() << std::endl;
  }

  return 0;
}
  

У меня есть 1000 udp_client-ов, которые отправляют 1000 запросов, используя данный порт: ./udp_client port amp;

например

./udp_client 5000 amp; ./udp_client 5001 amp; … ./udp_client 6000

Я хочу обрабатывать все запросы одновременно, нужно ли запускать каждый сервер в отдельном потоке? Если это правда, достаточно одного экземпляра io_service или мне нужно создать экземпляр для каждого потока.

Спасибо.

Комментарии:

1. Существует много подходов. 1) обрабатывать каждый запрос последовательно. 2) обрабатывать каждый запрос в отдельном процессе. 3) обрабатывать каждый запрос в отдельном потоке. 4) используйте пул потоков или процессов для обработки нескольких запросов каждый. 5) используйте конечный автомат для обработки нескольких запросов в одном потоке небольшими порциями. 6) многое другое..

2. Я хочу использовать поток для каждого соединения. В этом случае достаточно одного экземпляра io_service или мне нужно создать экземпляр для каждого потока?

3. Достаточно одного экземпляра io_service. Я рекомендую пул потоков для того, что вы хотите, поскольку вы, вероятно, на самом деле не хотите создавать 1000 потоков, а какое-то независимое число, которым вы можете управлять. Просто вызовите io_service::run() из всех рабочих потоков, и они получат долю событий.

4. Спасибо за ваш ответ, есть ли образец для этого предложения в документации boost?