目录
1、概述
muduo网络库给用户提供了两个主要的类
- TcpServer:用于编写服务器程序的
- TcpClient:用于编写客户端程序的
moduo库采用epoll+线程池
好处:能够把网络I/O( 用户的连接和断开 )的代码和业务(用户的可读写事件等)代码区分开
2、基于muduo网络库开发服务器程序步骤
1.组合TcpServer对象
2.创建EventLoop事件循环对象的指针
3.明确TcpServer构造函数需要什么参数,输出ChatServer的构造函数
4.在当前服务器类的构造函数当中,注册处理连接的回调函数和处理读写事件的回调函数
5.设置合适的服务端线程数量,muduo库会自己分配I/O线程和worker线程
3、代码演示
#include <muduo/net/TcpServer.h>
#include <muduo/net/EventLoop.h>
#include <iostream>
#include <functional> //绑定器
using namespace std;
using namespace muduo;
using namespace muduo::net;
using namespace placeholders;
/*基于muduo网络库开发服务器程序
1.组合TcpServer对象
2.创建EventLoop事件循环对象的指针
3.明确TcpServer构造函数需要什么参数,输出ChatServer的构造函数
4.在当前服务器类的构造函数当中,注册处理连接的回调函数和处理读写事件的回调函数
5.设置合适的服务端线程数量,muduo库会自己分配I/O线程和worker线程
*/
class ChatServer
{
public:
ChatServer(EventLoop *loop, // 事件循环
const InetAddress &listenAddr, // IP+Port
const string &nameArg) // 服务器的名字
: _server(loop, listenAddr, nameArg), _loop(loop)
{
// 给服务器注册用户连接的创建和断开回调
// 当监听到有创建和断开连接的请求时,调用onConnection方法
_server.setConnectionCallback(std::bind(&ChatServer::onConnection, this, _1)); //_1参数占位符,命名空间placeholders,onConnection只有一个参数
// 给服务器注册用户读写事件回调
_server.setMessageCallback(std::bind(&ChatServer::onMessage, this, _1, _2, _3)); //_1,_2,_3,onMessage有3个参数
// 设置服务器端的线程数量 1个I/O线程 3个worker线程
_server.setThreadNum(4);
}
// 开启事件循环
void start()
{
_server.start();
}
private:
// 专门处理用户的连接创建和断开 epoll listenfd accept
void onConnection(const TcpConnectionPtr &conn)
{
if(conn->connected())//连接成功返回true
{
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();//退出epoll,退出服务器
}
}
// 专门处理用户的读写事件
void onMessage(const TcpConnectionPtr &conn, // 连接
Buffer *buffer, // 缓冲区
Timestamp time) // 接收到数据的时间信息
{
string buf=buffer->retrieveAllAsString();
cout<<"recv data:"<<buf<<"time:"<<time.toString()<<endl;
conn->send(buf);
}
TcpServer _server; // #1
EventLoop *_loop; // #2 epoll
};
int main()
{
EventLoop loop;//epoll
InetAddress addr("127.0.0.1",6000);
ChatServer server(&loop,addr,"ChatServer");
server.start();//listenfd epoll_ctl=>epoll
loop.loop();//epoll_wait以阻塞方式等待新用户连接,已连接用户的读写事件等
return 0;
}