Boost.Asio翻译(三)--DayTime1

时间:2022-09-09 14:16:50
A synchronous TCP daytime client  

 一个同步的TCP daytime客户端

// client.cpp
//
// Copyright (c) 2003-2008 Christopher M. Kohlhoff (chris at kohlhoff dot com)
//
// Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)

#include <iostream>
#include <boost/array.hpp>
#include <boost/asio.hpp>
using boost::asio::ip::tcp;
int main(int argc, char* argv[])
{
  try
  {
    if (argc != 2)
    {
      std::cerr << "Usage: client <host>" << std::endl;
      return 1;
    }
    boost::asio::io_service io_service;
    tcp::resolver resolver(io_service);
    tcp::resolver::query query(argv[1], "daytime");
    tcp::resolver::iterator endpoint_iterator = resolver.resolve(query);
    tcp::resolver::iterator end;

    tcp::socket socket(io_service);
    boost::system::error_code error = boost::asio::error::host_not_found;
    while (error && endpoint_iterator != end)
    {
      socket.close();
      socket.connect(*endpoint_iterator++, error);
    }
    if (error)
      throw boost::system::system_error(error);

    for (;;)
    {
      boost::array<char, 128> buf;
      boost::system::error_code error;

      size_t len = socket.read_some(boost::asio::buffer(buf), error);

      if (error == boost::asio::error::eof)
        break; // Connection closed cleanly by peer.
      else if (error)
        throw boost::system::system_error(error); // Some other error.

      std::cout.write(buf.data(), len);
    }
  }
  catch (std::exception& e)
  {
    std::cerr << e.what() << std::endl;
  }
  return 0;
}

说明:

我们需要把服务器的名称转化为TCP的节点,而该名称是通过应用程序的参数指定的。我们使用boost::asio::ip::tcp::resolver 对象来完成。

 
 
Boost.Asio翻译(三)--DayTime1    tcp::resolver resolver(io_service);

一个resolver对象获得一个query对象,并将其转换为节点列表.我们通过argv[1]中的服务器名称服务名,在这里是daytime,构造一个query 

 
 
Boost.Asio翻译(三)--DayTime1   tcp::resolver::query query(argv[ 1 ],  " daytime " );

节点列表用 boost::asio::ip::tcp::resolver::iterator 类型的迭代器返回。返回的iterator将采用boost::asio::ip::tcp::resolver::iterator的默认构造函数来构造。

 
 
    tcp::resolver::iterator endpoint_iterator  =  resolver.resolve(query);
    tcp::resolver::iterator end;

现在我们建立一个socket并连接之,由于获得的节点既有IPv4也有IPv6的。所以,我们需要依次尝试访问它们直到找到一个可以正常工作的。这样做可使得我们的程序独立于特定的IP版本。

 
 
    tcp::socket socket(io_service);
    boost::system::error_code error 
=  boost::asio::error::host_not_found;
    
while  (error  &&  endpoint_iterator  !=  end)
    {
      socket.close();
      socket.connect(
* endpoint_iterator ++ , error);
    }
    
if  (error)
      
throw  boost::system::system_error(error);

连接打开后,现在我们需要做的就是读取daytime服务器的响应。

我们使用boost::array 来存放接收到的数据。boost::asio::buffer()函数会自动确定array的长度来防止缓冲区溢出。我们也可以使用 char [] 或 std::vector来代替boost::array。

 
 
     for  (;;)
    {
      boost::array
< char 128 >  buf;
      boost::system::error_code error;

      size_t len 
=  socket.read_some(boost::asio::buffer(buf), error);

当服务器关闭连接时,boost::asio::ip::tcp::socket::read_some()函数boost::asio::error::eof错误标志返回通过该错误标志,我们知道应该退出循环了

 
 
       if  (error  ==  boost::asio::error::eof)
        
break //  Connection closed cleanly by peer.
       else   if  (error)
        
throw  boost::system::system_error(error);  //  Some other error.

      std::cout.write(buf.data(), len);
    }