boost::asio::async_read зацикливается на ноль с нулевым байтом полученных данных

Я пытаюсь написать асинхронный TCP-клиент на основе IO_Service, в котором Async_write работает нормально, но async_read работает в бесконечном цикле. Во время моей попытки исправить проблему я обнаружил, что во всех других случаях async_read просто прекращает получать данные, ничего не получает, пока я не остановлю сервер. Ниже приведен код и ссылки, которые я пробовал перед отправкой запроса.

Предложения, которые я попробовал, точно такие же, как у меня, 2, 3, но во всех случаях мой обработчик async_read ничего не читает. В одном-единственном случае он запускает бесконечный цикл, когда я устанавливаю буфер как boost::asio::mutable_buffer bytes; в остальных случаях я пытался boost::array<char, 512> bytes;, boost::asio::streambuf bytes; а также char bytes[512]; где обработчик async_read не вызывается.

Проходя через все эти решения, я запутался: может ли это быть проблемой буфера? Нужно ли инициализировать его перед передачей для чтения?

Пожалуйста, руководство.

ScalableSocket::ScalableSocket()
{
    //ctor

    using namespace boost::asio;
    service = boost::make_shared<io_service>();
    work =    boost::make_shared<io_service::work>(*service );
    strand  = boost::make_shared<io_service::strand>( *service );
    worker_threads = boost::make_shared<boost::thread_group>();



        worker_threads->create_thread(boost::bind(&ScalableSocket::WorkerThread,this));


    resolver = boost::make_shared<boost::asio::ip::tcp::resolver> (*service);

    tcp_socket= boost::make_shared<boost::asio::ip::tcp::socket> (*service);

    boost::asio::ip::tcp::resolver::query q(boost::asio::ip::tcp::v4(),"192.168.100.96","9602");

    boost::asio::ip::tcp::resolver::iterator it = resolver->resolve(q);

    boost::asio::async_connect(*tcp_socket,it,boost::bind(&ScalableSocket::connect_handler,this,boost::asio::placeholders::error));

    tcp_socket->set_option(boost::asio::ip::tcp::no_delay(true) );

}

ScalableSocket::~ScalableSocket()
{
    //dtor

}

void ScalableSocket::PublishPost()
{
    strand->post(boost::bind(&ScalableSocket::OnSend,this));
}


void ScalableSocket::OnSend()
{


    boost::array<char, 6> a = { 'a', 'b', 'c', 'd', 'e' };

    boost::asio::async_write(*tcp_socket,boost::asio::buffer(a),
                                    boost::bind(&ScalableSocket::write_handler, this, boost::asio::placeholders::error,boost::asio::placeholders::bytes_transferred));



}

void ScalableSocket::WorkerThread()
{
    while( true )
    {
        try
        {
            boost::system::error_code ec;
            service->run( ec );
            if( ec )
            {
                ///LOGE(ec);
            }
            break;
        }
        catch( std::exception & ex )
        {
            ///LOGE(ex.what());
        }
    }
}

void ScalableSocket::connect_handler(const boost::system::error_code &ec)
{
    if (!ec)
    {

    PublishPost();




/* boost::asio::async_read(*tcp_socket,
                                boost::asio::buffer(bytes),
                                boost::bind(&ScalableSocket::read_handler, this,
                                            boost::asio::placeholders::error,boost::asio::placeholders::bytes_transferred));
*/

        ///https://stackru.com/questions/4527443/problems-using-boostasioasync-read

        boost::shared_ptr<boost::array<char, 512>> buf(new boost::array<char, 512>);


        boost::asio::async_read(*tcp_socket,boost::asio::buffer(*buf),
                                boost::bind(&ScalableSocket::read_handler, this,buf,
                                boost::asio::placeholders::error,boost::asio::placeholders::bytes_transferred));

    }
    else
    {
        cout<<" Some error connecting to Exchange  "<< ec.message()<<endl;

    }

}


void ScalableSocket::OnTimer(const boost::system::error_code &ec)
{
    if(!ec)
    {
        printf("\n\n Heartbeat event raised sending KeepAlive to exchange \n\n");
        PublishPost();
        HeartBeatTimer->async_wait(boost::bind(&ScalableSocket::OnTimer,this, boost::asio::placeholders::error));
    }
}

void ScalableSocket::recvData()
{
    boost::system::error_code error;
    boost::array<char, 1024> buf;

    //for(;;)
    {
        size_t len = tcp_socket->read_some(boost::asio::buffer(buf), error);

        cout<<"\n Recv data size is "<<len;

    }
}

void ScalableSocket::read_handler(boost::shared_ptr<boost::array<char, 512>> buf,const boost::system::error_code &ec,std::size_t bytes_transferred)
{


    if (!ec )//&& bytes_transferred > 0)
    {

        ///recvData(); /// If i enable this code during infinite loop it start getting data that means socket has no issue

        cout << " Data size recieved "<< bytes_transferred<<endl;




            boost::asio::async_read(*tcp_socket,boost::asio::buffer(*buf),
                                boost::bind(&ScalableSocket::read_handler, this,buf,
                                boost::asio::placeholders::error,boost::asio::placeholders::bytes_transferred));
    }
    else
    {
        /// Some issue with socket publish error , inform user and reconnect
        cout<<" Some error reading data from Exchange  "<< ec.message()<<endl;

    }
}

void ScalableSocket::write_handler(const boost::system::error_code& error,std::size_t bytes_transferred)
{
    if(!error)
    {
        /// data Sent successfully
        cout<< " Data sent size "<< bytes_transferred<<endl;

    }
    else
    {
        cout<<" Some error sending data to Exchange  "<< error.message()<<endl;
    }

}

1 ответ

Решение

asnyc_readНЕ будет "возвращать" / вызывать обработчик, пока данный буфер не будет полностью заполнен.

asnyc_read_some вернется после прочтения нескольких байтов. Это может быть функция, которую вы ищете.

Не забудьте правильно обрабатывать полученные данные с помощью asnyc_read_some, Если вы отправите 512 байт, это может произойти за пару операций чтения, в зависимости от компьютера.

Другие вопросы по тегам