见:http://www.boost.org/doc/libs/1_61_0/doc/html/boost_asio/examples/cpp11_examples.html
或者:https://github.com/NearXdu/AsioLearn
1.服务端
对于一个广播聊天室来说。
服务端能够接受连接,并且将收到的消息转发到所有已连接的客户端。
因此,在设计服务端代码时,需要考虑如何保存Tcp连接实例。
在Boost.Asio中给出的例子中一共设计了3个类(实际上是4个,其中一个作为虚基类)。
1. Server
2. Room
3. Session
Server类对象异步的接收连接,它包含一个Room类对象引用,将管理所有的连接(Session对象的shared_ptr)当Server收到消息之后,将向所有的Session(连接)转发消息。
#include <cstdlib>
#include <deque>
#include <iostream>
#include <list>
#include <memory>
#include <set>
#include <utility>
#include <boost/asio.hpp>
#include "chat_message.hpp"
using boost::asio::ip::tcp;
typedef std::deque<chat_message> chat_message_queue;
class chat_participant
{
public:
virtual ~chat_participant() {}
virtual void deliver(const chat_message& msg) = 0; //participant should deliver message
};
typedef std::shared_ptr<chat_participant> chat_participant_ptr;//shared ptr
class chat_room
{
public:
void join(chat_participant_ptr participant)
{
participants_.insert(participant);//add a client
//将之前的消息写一遍给最新的连接者
for (auto msg: recent_msgs_)
participant->deliver(msg);
}
void leave(chat_participant_ptr participant)
{
participants_.erase(participant);//remove a client
}
void deliver(const chat_message& msg)
{
//先将发送的消息放入待发送列表。
recent_msgs_.push_back(msg);
while (recent_msgs_.size() > max_recent_msgs)
recent_msgs_.pop_front();//delete expired message
for (auto participant: participants_)
participant->deliver(msg);//将消息向所有的对象转发一遍
}
private:
std::set<chat_participant_ptr> participants_;//all client
enum { max_recent_msgs = 100 };
chat_message_queue recent_msgs_;
};//chat r