http://blog.csdn.net/cometeor/article/details/4901332
以下是来自boost example的异步echo server:
- //
- // async_tcp_echo_server.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 <cstdlib>
- #include <iostream>
- #include <boost/bind.hpp>
- #include <boost/asio.hpp>
- using boost::asio::ip::tcp;
- class session
- {
- public:
- session(boost::asio::io_service& io_service)
- : socket_(io_service)
- {
- }
- tcp::socket& socket()
- {
- return socket_;
- }
- void start()
- {
- socket_.async_read_some(boost::asio::buffer(data_, max_length),
- boost::bind(&session::handle_read, this,
- boost::asio::placeholders::error,
- boost::asio::placeholders::bytes_transferred));
- }
- void handle_read(const boost::system::error_code& error,
- size_t bytes_transferred)
- {
- if (!error)
- {
- boost::asio::async_write(socket_,
- boost::asio::buffer(data_, bytes_transferred),
- boost::bind(&session::handle_write, this,
- boost::asio::placeholders::error));
- }
- else
- {
- delete this;
- }
- }
- void handle_write(const boost::system::error_code& error)
- {
- if (!error)
- {
- socket_.async_read_some(boost::asio::buffer(data_, max_length),
- boost::bind(&session::handle_read, this,
- boost::asio::placeholders::error,
- boost::asio::placeholders::bytes_transferred));
- }
- else
- {
- delete this;
- }
- }
- private:
- tcp::socket socket_;
- enum { max_length = 1024 };
- char data_[max_length];
- };
- class server
- {
- public:
- server(boost::asio::io_service& io_service, short port)
- : io_service_(io_service),
- acceptor_(io_service, tcp::endpoint(tcp::v4(), port))
- {
- session* new_session = new session(io_service_);
- acceptor_.async_accept(new_session->socket(),
- boost::bind(&server::handle_accept, this, new_session,
- boost::asio::placeholders::error));
- }
- void handle_accept(session* new_session,
- const boost::system::error_code& error)
- {
- if (!error)
- {
- new_session->start();
- new_session = new session(io_service_);
- acceptor_.async_accept(new_session->socket(),
- boost::bind(&server::handle_accept, this, new_session,
- boost::asio::placeholders::error));
- }
- else
- {
- delete new_session;
- }
- }
- private:
- boost::asio::io_service& io_service_;
- tcp::acceptor acceptor_;
- };
- int main(int argc, char* argv[])
- {
- try
- {
- if (argc != 2)
- {
- std::cerr << "Usage: async_tcp_echo_server <port>/n";
- return 1;
- }
- boost::asio::io_service io_service;
- using namespace std; // For atoi.
- server s(io_service, atoi(argv[1]));
- io_service.run();
- }
- catch (std::exception& e)
- {
- std::cerr << "Exception: " << e.what() << "/n";
- }
- return 0;
- }
为了模拟高并发连接,顺手写了个异步多连接的客户端程序:
- #include <cstdlib>
- #include <cstring>
- #include <iostream>
- #include <boost/asio.hpp>
- #include <boost/bind.hpp>
- using boost::asio::ip::tcp;
- using namespace std;
- static int id = 1;
- const char message[] = "test write string...";
- class echo_session
- {
- public:
- echo_session(boost::asio::io_service& io_service)
- : socket_(io_service)
- {
- id_ = id;
- ++id;
- }
- void start(const std::string& ip, const std::string& port)
- {
- //解析主机地址
- tcp::resolver resolver(socket_.io_service());
- tcp::resolver::query query(tcp::v4(), ip, port);
- tcp::resolver::iterator iterator = resolver.resolve(query);
- //异步连接
- socket_.async_connect(*iterator, boost::bind(&echo_session::handle_connect, this, boost::asio::placeholders::error));
- }
- private:
- void handle_connect(const boost::system::error_code& error)
- {
- if (!error)
- {
- //连接成功,发送message中的数据
- boost::asio::async_write(socket_,
- boost::asio::buffer(message, sizeof(message)),
- boost::bind(&echo_session::handle_write, this,
- boost::asio::placeholders::error));
- }
- else
- cout << error << endl;
- }
- void handle_write(const boost::system::error_code& error)
- {
- if (!error)
- {
- //写入完毕,接收服务器回射的消息
- boost::asio::async_read(socket_, boost::asio::buffer(buf_, sizeof(buf_)),
- boost::bind(&echo_session::handle_read, this,
- boost::asio::placeholders::error, boost::asio::placeholders::bytes_transferred));
- }
- else
- cout << error << endl;
- }
- void handle_read(const boost::system::error_code& error, size_t bytes_transferred)
- {
- if (!error)
- {
- //读取完毕,在终端显示
- cout << id_ << ":receive:" << bytes_transferred << "," << buf_ << endl;
- //周而复始...
- handle_connect(error);
- }
- else
- cout << error << endl;
- }
- int id_;
- tcp::socket socket_;
- char buf_[sizeof(message)];
- };
- int main(int argc, char* argv[])
- {
- const int session_num = 10000; //连接的数量
- echo_session* sessions[session_num];
- memset(sessions, 0, sizeof(sessions));
- try
- {
- if (argc != 3)
- {
- std::cerr << "Usage: blocking_tcp_echo_client <host> <port>/n";
- return 1;
- }
- boost::asio::io_service io_service;
- //创建session_num个连接
- for (int i=0; i<session_num; ++i)
- {
- sessions[i] = new echo_session(io_service);
- sessions[i]->start(argv[1], argv[2]);
- }
- //io_service主循环
- io_service.run();
- for (int i=0; i<session_num; ++i)
- if (sessions[i] != NULL)
- delete sessions[i];
- }
- catch (std::exception& e)
- {
- for (int i=0; i<session_num; ++i)
- if (sessions[i] != NULL)
- delete sessions[i];
- std::cerr << "Exception: " << e.what() << "/n";
- }
- return 0;
- }
在linux下操作系统默认允许的最大连接数可能不足导致连接抛异常,需要用ulimit -n命令修改。如果提示操作不允许,需要修改系统配置文件。详细