muduo网络库简介
muduo网络库给用户提供了两个主要的类
TcpServer : 用户编写服务器程序的
TcpClient : 用户编写客户端程序的
优点:
epoll + 线程池
好处 : 能够把网络I/O的代码和业务代码区分开(用户的连接和断开 ,用户的可读写事件)
基于muduo网络库开发服务器流程
编写一个ChatServer为例:
- 组合TcpServer对象
- 创建EventLoop事件循环对象的指针
- 明确TcpServer构造函数需要哪些参数,输出ChatServer的构造函数
- 在当前服务器类的构造函数中,注册处理连接的回调函数和处理读写事件的回调函数
- 设置合适的服务端线程数量,muduo库会自己分配I/O线程和worker线程
#include <muduo/net/TcpServer.h>
#include <muduo/net/EventLoop.h>
#include <iostream>
#include <functional>
#include <string>
using namespace std;
using namespace muduo;
using namespace muduo::net;
using namespace placeholders;
class ChatServer {
public:
ChatServer(EventLoop* loop, // 事件循环
const InetAddress& listenAddr, // IP + Port
const string& nameArg) // 服务器名字
: _server(loop, listenAddr, nameArg), _loop(loop)
{
// 给服务器注册用户连接和创建和断开回调
_server.setConnectionCallback(std::bind(&ChatServer::onConnection, this, _1));
// 给服务器注册用户读写事件回调
_server.setMessageCallback(std::bind(&ChatServer::onMessage, this, _1, _2, _3));
// 设置服务端线程数量 1个I/O线程,3个worker线程
_server.setThreadNum(4);
}
// 开启事件循环
void start() {
_server.start();
}
private:
TcpServer _server; // #1
EventLoop* _loop; // #2 epoll
// 专门处理用户的连接创建和断开 epoll listenfd accept
void onConnection(const TcpConnectionPtr& conn) {
if (conn->connected()) {
cout << conn->peerAddress().toIpPort() << " -> " <<
conn->localAddress().toIpPort() <<" state: online " << endl;
}
else {
cout << conn->peerAddress().toIpPort() << " -> " <<
conn->localAddress().toIpPort() <<" state: offline " << endl;
conn->shutdown(); // close(fd)
// _loop->quit();
}
}
// 专门处理用户的读写事件
void onMessage(const TcpConnectionPtr& conn, // 连接
Buffer* buffer, // 缓冲区
Timestamp time) // 收到数据的时间信息
{
string buf = buffer->retrieveAllAsString();
cout << "recv data: " << buf << "time: " << time.toString() << endl;
conn->send(buf);
}
};
必须要指定构造函数,因为默认的需要四个参数
构造函数中编写回调函数,利用绑定器和专门处理事件的函数(和参数)绑定。
编译并链接相应的库
使用telnet检测连接
现在可以进行数据传输
退出示例: