// 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 对象来完成。
tcp::resolverresolver(io_service);
一个resolver对象获得一个query对象,并将其转换为节点列表.我们通过argv[1]中的服务器名称和服务名,在这里是daytime,构造一个query。
tcp::resolver::queryquery(argv[
1
],
"
daytime
"
);
节点列表用boost::asio::ip::tcp::resolver::iterator类型的迭代器返回。返回的iterator将采用boost::asio::ip::tcp::resolver::iterator的默认构造函数来构造。
tcp::resolver::iteratorendpoint_iterator
=
resolver.resolve(query);
tcp::resolver::iteratorend;
现在我们建立一个socket并连接之,由于获得的节点既有IPv4也有IPv6的。所以,我们需要依次尝试访问它们直到找到一个可以正常工作的。这样做可使得我们的程序独立于特定的IP版本。
tcp::socketsocket(io_service);
boost::system::error_codeerror
=
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_codeerror;
size_tlen
=
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
;
//
Connectionclosedcleanlybypeer.
else
if
(error)
throw
boost::system::system_error(error);
//
Someothererror.
std::cout.write(buf.data(),len);
}