c10k测试:使用asio实现多线程回显…

      asio是boost库里面example和文档做得比较好的一个库了。里面自带了一个echo server,不过可惜是单线程的,没办法充分发挥多核主机的威力。于是我把它改造成多线程的,主要是抄了http server example 里io_service_pool的代码嘿嘿。
      测试的结果当然不至于比每连接开线程的差,100-15000连接时也都把带宽跑满了。
============================代码的分隔线==========================
#include <cstdlib>
#include <iostream>
#include <stdexcept>
#include <boost/bind.hpp>
#include <boost/asio.hpp>
#include <boost/lexical_cast.hpp>
#include <boost/thread.hpp>

using namespace std;
using boost::asio::ip::tcp;

class io_service_pool
      : public boost::noncopyable
{
public:

      explicit io_service_pool(std::size_t pool_size)
            : next_io_service_(0)
                     
            for (std::size_t i = 0; i < pool_size; ++ i)
            {
                  io_service_sptr io_service(new boost::asio::io_service);
                  work_sptr work(new boost::asio::io_service::work(*io_service));
                  io_services_.push_back(io_service);
                  work_.push_back(work);
            }
      }

      void start()
               
            for (std::size_t i = 0; i < io_services_.size(); ++ i)
            {
                  boost::shared_ptr<boost::thread> thread(new boost::thread(
                        boost::bind(&boost::asio::io_service::run, io_services_[i])));
                  threads_.push_back(thread);
            }
      }

      void join()
      {
            for (std::size_t i = 0; i < threads_.size(); ++ i)
            {
                  threads_[i]->join();
                     
      }

      void stop()
               
            for (std::size_t i = 0; i < io_services_.size(); ++ i)
            {
                  io_services_[i]->stop();
            }
      }

      boost::asio::io_service& get_io_service()
      {
            boost::mutex::scoped_lock lock(mtx);
            boost::asio::io_service& io_service = *io_services_[next_io_service_];
            ++ next_io_service_;
            if (next_io_service_ == io_services_.size())
            {
                  next_io_service_ = 0;
            }
            return io_service;
      }

private:
      typedef boost::shared_ptr<boost::asio::io_service> io_service_sptr;
      typedef boost::shared_ptr<boost::asio::io_service::work> work_sptr;
      typedef boost::shared_ptr<boost::thread> thread_sptr;

      boost::mutex mtx;

      std::vector<io_service_sptr> io_services_;
      std::vector<work_sptr> work_;
      std::vector<thread_sptr> threads_;           
      std::size_t next_io_service_;
};

boost::mutex cout_mtx;
int packet_size = 0;
enum {MAX_PACKET_LEN = 4096};

class session
{
public:
      session(boost::asio::io_service& io_service)
            : socket_(io_service)
            , recv_times(0)
      {
      }
     
      virtual ~session()
      {
            boost::mutex::scoped_lock lock(cout_mtx);
            cout << recv_times << endl;
      }

      tcp::socket& socket()
      {
            return socket_;
      }

      inline void start()
      {
            boost::asio::async_read(socket_, boost::asio::buffer(data_, packet_size),
                  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)
            {
                  ++ recv_times;
                  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)
            {
                  start();
            }
            else
            {
                  delete this;
            }
      }

private:
      tcp::socket socket_;
      char data_[MAX_PACKET_LEN];
      int recv_times;
};

class server
{
public:
      server(short port, int thread_cnt)
            : io_service_pool_(thread_cnt)
            , acceptor_(io_service_pool_.get_io_service(), tcp::endpoint(tcp::v4(), port))
      {
            session* new_session = new session(io_service_pool_.get_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_pool_.get_io_service());
                  acceptor_.async_accept(new_session->socket(),
                        boost::bind(&server::handle_accept, this, new_session, boost::asio::placeholders::error));
            }
            else
            {
                  delete new_session;
            }
      }

      void run()
      {
            io_service_pool_.start();
            io_service_pool_.join();
      }

private:
     
      io_service_pool io_service_pool_;
      tcp::acceptor acceptor_;
};

int main(int argc, char* argv[])
{
      try
      {
            if (argc != 4)
            {
                  cerr << "Usage: async_tcp_echo_server <port> <packet_size> <thread_cnt>" << endl;
                  return 1;
            }
            short port = boost::lexical_cast<short>(argv[1]);
            packet_size = boost::lexical_cast<int>(argv[2]);
            int thread_cnt = boost::lexical_cast<int>(argv[3]);
            if (packet_size <= 0 || packet_size > MAX_PACKET_LEN)
            {
                  cerr << "packet_size <= 0 || packet_size > MAX_PACKET_LEN" << endl;
                  return 1;
            }
            if (thread_cnt <= 0)
            {
                  cerr << "thread_cnt <= 0" << endl;
                  return 1;                 
                     

            server s(port, thread_cnt);
            s.run();
      }
      catch (std::exception& e)
      {
            cerr << "Exception: " << e.what() << endl;
            return 1;
      }

      return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值