boost.asio简单入坑

摘要:
目前,银行接口模块中的Bu使用短连接,短连接使用boost Asio库。接下来,我们将简要介绍如何入坑。让我们举一个客户端示例:#defineBOOST_RECEX_NO_LIB#include无效_ cli1(){boost::asio::io_serviceservice;//在堆栈中定义io_service类的对象,将自动调用io_service类的构造函数。//from_string将字符串格式ip转换为硬件识别的地址t对象,并调用构造函数basic_stream_socketsock。连接size_tlen=袜子。邮寄soc k.close();}它比直接调用系统接口更容易使用,并且支持跨平台。示例源代码来自官方文档:http://www.boost.org/doc/libs/1_50_0/doc/html/boost_asio/example/allocation/server.cpp其中_ 50_ 0是版本号,可以修改为自己使用的版本号。每个版本下都有相应的示例。

网络连接有短连接和长连接。顾名思义,connect之后一直不断开的是长连接,像公司用的数据Route模块;三次握手建立连接后收发一次数据后就断开,断开是四次握手,再次收发数据就再来三次握手建立连接是短连接。每种模式都有优缺点。短连接优点是:如果客户端数量大的话,可以让每个客户轮流请求到数据。缺点:每次连接(三次握手)断开(四次握手)都耗费资源。长连接相对的优点:省资源。缺点:客户量大就会有很多连接不上。各自的用途:长连接适合I/O频繁型的,有很多数据需要传输的,且连接数不需要太多的,像数据Route模块。短连接适合于网站,http服务,比如apache。目前银行接口模块中有bu采用的是短连接,它使用了boost.asio库,接下来简单介绍下如何入坑。

先来个客户端的例子:

#define BOOST_REGEX_NO_LIB
#include <boost/asio.hpp>

void asio_cli1() {
	boost::asio::io_service service;// 在栈中定义类io_service的对象,会自动调用类io_service的构造函数
	// from_string会把字符串格式ip转成硬件识别的地址
	// 类似于sockaddr_in.sin_addr.S_un.S_addr = inet_addr("10.243.141.16");
	boost::asio::ip::tcp::endpoint ep(boost::asio::ip::address::from_string("10.243.141.16"), 9000);
	
	boost::asio::ip::tcp::socket sock(service);// 定义socket对象并调用了构造函数basic_stream_socket(boost::asio::io_service& io_service)
	
	sock.connect(ep);
	size_t len = sock.send(boost::asio::buffer("hello asio
"));
	sock.close();

}

  使用上比直接调用系统接口方便很多,还支持跨平台。然后我们可以在buffer中的数据换成各家银行格式的报文就可以模拟银行端的发起请求了。

      服务器端的例子:

#define BOOST_REGEX_NO_LIB
#include <boost/asio.hpp>
using namespace std;

void asio_svr() {
	try {
		boost::asio::io_service io;
		//boost::asio::ip::tcp::endpoint ep(boost::asio::ip::address::from_string("10.253.41.49"), 9000);
		//boost::asio::ip::tcp::acceptor acceptor(io, ep);
		boost::asio::ip::tcp::acceptor acceptor(io, boost::asio::ip::tcp::endpoint(boost::asio::ip::tcp::v4(), 9000));
		cout<<acceptor.local_endpoint().address()<<endl;
		char buf[1024] = {0};
		for( ; ; ) {
			boost::asio::ip::tcp::socket sock(io);
			acceptor.accept(sock);
			cout<<"cli:"<<sock.remote_endpoint().address()<<endl;
			sock.receive(boost::asio::buffer(buf));
			cout<<buf<<endl;

		}
	} 
	catch (std::exception& e) {
        std::cerr << e.what() << std::endl;
    }

}

  接下来整个复杂点的,异步模式,IOCP模型(I/O Completion Port)。例子源码来自于官方文档:http://www.boost.org/doc/libs/1_50_0/doc/html/boost_asio/example/allocation/server.cpp

其中1_50_0是版本号,可以修改成自己用的那个,每个版本下都有对应例子的。

#include <cstdlib>
#include <iostream>
#include <boost/aligned_storage.hpp>
#include <boost/array.hpp>
#include <boost/bind.hpp>
#include <boost/enable_shared_from_this.hpp>
#include <boost/noncopyable.hpp>
#include <boost/shared_ptr.hpp>
#include <boost/asio.hpp>

using boost::asio::ip::tcp;

// Class to manage the memory to be used for handler-based custom allocation.
// It contains a single block of memory which may be returned for allocation
// requests. If the memory is in use when an allocation request is made, the
// allocator delegates allocation to the global heap.
class handler_allocator
  : private boost::noncopyable
{
public:
  handler_allocator()
    : in_use_(false)
  {
  }

  void* allocate(std::size_t size)
  {
    if (!in_use_ && size < storage_.size)
    {
      in_use_ = true;
      return storage_.address();
    }
    else
    {
      return ::operator new(size);
    }
  }

  void deallocate(void* pointer)
  {
    if (pointer == storage_.address())
    {
      in_use_ = false;
    }
    else
    {
      ::operator delete(pointer);
    }
  }

private:
  // Storage space used for handler-based custom memory allocation.
  boost::aligned_storage<1024> storage_;

  // Whether the handler-based custom allocation storage has been used.
  bool in_use_;
};

// Wrapper class template for handler objects to allow handler memory
// allocation to be customised. Calls to operator() are forwarded to the
// encapsulated handler.
template <typename Handler>
class custom_alloc_handler
{
public:
  custom_alloc_handler(handler_allocator& a, Handler h)
    : allocator_(a),
      handler_(h)
  {
  }

  template <typename Arg1>
  void operator()(Arg1 arg1)
  {
    handler_(arg1);
  }

  template <typename Arg1, typename Arg2>
  void operator()(Arg1 arg1, Arg2 arg2)
  {
    handler_(arg1, arg2);
  }

  friend void* asio_handler_allocate(std::size_t size,
      custom_alloc_handler<Handler>* this_handler)
  {
    return this_handler->allocator_.allocate(size);
  }

  friend void asio_handler_deallocate(void* pointer, std::size_t /*size*/,
      custom_alloc_handler<Handler>* this_handler)
  {
    this_handler->allocator_.deallocate(pointer);
  }

private:
  handler_allocator& allocator_;
  Handler handler_;
};

// Helper function to wrap a handler object to add custom allocation.
template <typename Handler>
inline custom_alloc_handler<Handler> make_custom_alloc_handler(
    handler_allocator& a, Handler h)
{
  return custom_alloc_handler<Handler>(a, h);
}

class session
  : public boost::enable_shared_from_this<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_),
        make_custom_alloc_handler(allocator_,
          boost::bind(&session::handle_read,
            shared_from_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),
          make_custom_alloc_handler(allocator_,
            boost::bind(&session::handle_write,
              shared_from_this(),
              boost::asio::placeholders::error)));
    }
  }

  void handle_write(const boost::system::error_code& error)
  {
    if (!error)
    {
      socket_.async_read_some(boost::asio::buffer(data_),
          make_custom_alloc_handler(allocator_,
            boost::bind(&session::handle_read,
              shared_from_this(),
              boost::asio::placeholders::error,
              boost::asio::placeholders::bytes_transferred)));
    }
  }

private:
  // The socket used to communicate with the client.
  tcp::socket socket_;

  // Buffer used to store data received from the client.
  boost::array<char, 1024> data_;

  // The allocator to use for handler-based custom memory allocation.
  handler_allocator allocator_;
};

typedef boost::shared_ptr<session> session_ptr;

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_ptr 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_ptr new_session,
      const boost::system::error_code& error)
  {
    if (!error)
    {
      new_session->start();
    }

    new_session.reset(new session(io_service_));
    acceptor_.async_accept(new_session->socket(),
        boost::bind(&server::handle_accept, this, new_session,
          boost::asio::placeholders::error));
  }

private:
  boost::asio::io_service& io_service_;
  tcp::acceptor acceptor_;
};

int main(int argc, char* argv[])
{
  try
  {
    if (argc != 2)
    {
      std::cerr << "Usage: server <port>
";
      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() << "
";
  }

  return 0;
}

  在服务端启动后调试,看到的是在io_service.run()处阻塞了,run函数内调用了while语句,就会一直监听有没有客户端连接。在windows下实现在win_iocp_io_service.ipp文件内。官方给你例子比较复杂,那么来个简单点的,参考了http://blog.csdn.net/zhuky/article/details/5364685

       写一个server类,代码如下:

#include <iostream>
#include <string>
#include <boost/asio.hpp>
#include <boost/bind.hpp>
#include <boost/smart_ptr.hpp>

using namespace boost::asio;
using boost::system::error_code;
using ip::tcp;

// 接收多个客户端连接 然后给客户端发送一条消息
// 如果发送成功就把发送的数据打印出来
class CAsyncService {
public:
    CAsyncService(io_service &iosev, size_t port)
     :m_iosrv(iosev), m_acceptor(iosev, tcp::endpoint(tcp::v4(), port))
    {
    }
    void Start()
    {
        // 开始等待连接(非阻塞)
        boost::shared_ptr<tcp::socket> psocket(new tcp::socket(m_iosrv));
        // 触发的事件只有error_code参数,所以用boost::bind把socket绑定进去
        m_acceptor.async_accept(*psocket,
         boost::bind(&CAsyncService::AcceptHandler, this, psocket, _1)
         );
    }
    // 有客户端连接时AcceptHandler触发
    void AcceptHandler(boost::shared_ptr<tcp::socket> psocket, error_code ec)
     {
        if(ec) return;
        // 继续等待连接
         Start();
        // 显示远程IP
         std::cout << psocket->remote_endpoint().address() << std::endl;
        // 发送信息(非阻塞)
         boost::shared_ptr<std::string> pstr(new std::string("Hello async world!"));
         psocket->async_write_some(buffer(*pstr),
             boost::bind(&CAsyncService::WriteHandler, this, pstr, _1, _2)
             );
     }
    // 异步写操作完成后WriteHandler触发
    void WriteHandler(boost::shared_ptr<std::string> pstr,
         error_code ec, size_t bytes_transferred)
     {
        if(ec)
             std::cout<< "发送失败!" << std::endl;
        else
             std::cout<< *pstr << " 已发送" << std::endl;
     }
private:
     io_service &m_iosrv;
     ip::tcp::acceptor m_acceptor;
};

再简单使用下main函数调用:

int main(int argc, char* argv[])
{
     io_service io;
     CAsyncService ayncserv(io, 9000);
     ayncserv.Start();
     io.run();// 开始等待连接
    return 0;
}

免责声明:文章转载自《boost.asio简单入坑》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇springboot 有关拦截器遇到的问题vue实现文件夹的上传和下载下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

深入浅出--iOS的TCP/IP协议族剖析&amp;amp;&amp;amp;Socket

深入浅出--iOS的TCP/IP协议族剖析&&Socket 简介 该篇文章主要回顾--TCP/IP协议族中的TCP/UDP、HTTP;还有Socket。(--该文很干,酝酿了许久!你能耐心看完吗?) 我在这个文章中,列举了常见的TCP/IP族中的协议,今天主角是--传输层协议。 传输层(Transport Layer)是OSI(七...

RocketMQ 入门

一、rocketMQ是什么 rocketmq是一款低延迟、高可靠、可伸缩、已使用的消息中间件。具有以下特性: 1、支持发布/订阅、点对点(p2p)消息模型 2、同一个队列中支持先进先出(FIFO)和严格的顺序传递 3、支持拉(pull)和推(push)两种消息模式 4、单一队列百万消息的堆积能力 5、支持多种消息协议,比如: JMS 、MQTT 6、分布式...

Nginx网络连接配置

本文记录Nginx中网络连接的设置配置项。 1)读取HTTP头部超时时间 语法:client_header_timeout time(默认单位:秒); 默认:client_header_timeout 60; 配置块: http、server、location 在与客户端建立连接之后,将开始获取HTTP头部,如果在配置的时间之内没有读取到客户端发来的字节,...

分布式系统(二) --SOA 以及一些网络通信协议TCP/UDP SYN攻击

SOA(面向服务的架构):Service Oriented Architecture面向服务的架构。也就是把工程拆分成服务层、表现层两个工程。服务层中包含业务逻辑,只需要对外提供服务即可。表现层只需要处理和页面的交互,业务逻辑都是调用服务层的服务来实现。这样做的好处就是,系统之间的调用很方便,A系统要用到B系统,直接调用B系统的服务层就可以了。 集群就是多...

Linux Linux内核参数调优

Linux内核参数调优 by:授客QQ:1033553122 关于调优的建议: 1、出错时,可以查看操作系统日志,可能会找到一些有用的信息 2、尽量不要“批量”修改内核参数,笔者就曾这么干过,结果“调优”后,性能反而下降,事务出错数反而增加,所以,调优的时候可以考虑逐个参数进行调优,然后对比效果。   说明:我也不是很懂,参考自网络整理了下可能需要调整...

Python之路【第七篇】python基础 之socket网络编程

本篇文章大部分借鉴http://www.cnblogs.com/nulige/p/6235531.htmlpython socket 网络编程一、服务端和客户端 BS架构 (腾讯通软件:server+client) CS架构 (web网站) C/S架构与socket的关系: 我们学习socket就是为了完成C/S架构的开发 二、OSI七层模型 互联网协议按...