C++项目——集群聊天服务器项目(三)muduo网络库

今天来介绍集群聊天器项目中网络模块代码的核心模块——muduo网络库,一起来看看吧~

环境搭建C++项目——集群聊天服务器项目(一)项目介绍、环境搭建、Boost库安装、Muduo库安装、Linux与vscode配置-CSDN博客

Json第三方库C++项目——集群聊天服务器项目(二)Json第三方库-CSDN博客

一、muduo网络库介绍

muduo由陈硕大佬开发,是一个基于非阻塞IO和事件驱动的C++高并发TCP网络库

网络设计:reactors in threads - one loop per thread

one loop per thread指的是:

(1)一个线程只能有一个事件循环(EventLoop)

(2)一个文件描述符只能由一个线程进行读写,即一个TCP连接必须归属于某个EventLoop管理。

方案的特点是one loop per thread,有一个main reactor负载accept连接,然后把连接分发到某个sub reactor,该连接的所用操作都在那个sub reactor所处的线程中完成。

多个连接可能被分派到多个线程中,以充分利用CPU。

 Reactor poll的大小是固定的,根据CPU的数目确定。

原理:一个Base IO thread负责accept新的连接,接收到新的连接以后,使用轮询的方式在reactor pool中找到合适的sub reactor将这个连接挂载上去,这个连接上的所有任务都在这个sub reactor上完成。

如果有过多的耗费CPU I/O的计算任务,可以提交到创建的ThreadPool线程池中专门处理耗时的计算任务。

将epoll和线程池封装起来,好处是能够把网络的I/O代码与业务代码区分开

二、muduo网络库主要的类

TcpServer:用于编写服务器程序的类

TcpClient:用于编写客户端程序的类

接下来使用muduo网络库开发一个基本的服务器程序

三、基于muduo网络库开发服务器程序

3.1 基本步骤

1.组合TcpServer对象

2.创建EventLoop事件循环对象的指针

3.明确TcpServer构造函数需要什么参数,输出ChatServer的构造函数

4.在当前服务器类的构造函数中,注册处理连接和读写事件的回调函数

5.设置合适的服务端线程数量,muduo会自动划分I/O线程和worker线程

3.2 代码

 头文件

#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;

组合TcpServer对象

创建EventLoop事件循环对象的指针

明确TcpServer构造函数需要什么参数

class ChatServer
{
public:
    ChatServer(EventLoop* loop,                 //事件循环——Reactor反应堆
            const InetAddress& listenAddr,      //IP和端口
            const string& nameArg)              //服务器的名字
              : _server(loop, listenAddr, nameArg), _loop(loop){}
private:
    TcpServer _server;
    EventLoop *_loop;       //epoll
};

输出ChatServer的构造函数

构造函数负责给服务器注册用户连接与断开回调函数,注册读写事件回调函数,并设置线程数量,muduo网络库会自动分配线程用于主reactor和子reactor

 ChatServer(EventLoop* loop,                 //事件循环——Reactor反应堆
            const InetAddress& listenAddr,      //IP和端口
            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();
    }

连接与断开回调函数:显示上线和下线

    //专门处理用户连接创建和断开    epoll
    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();
        }
    }

读写事件回调函数:这里的功能是将客户端发来的信息发回去

    //专门处理用户读写事件    
    void OnMessage(const TcpConnectionPtr&conn ,    //连接
                            Buffer*buffer,            //缓冲区
                            Timestamp time)          //接受到数据的时间信息
    {         
        string buf = buffer->retrieveAllAsString();
        cout << "recv data : " << buf << " time:" << time.toString() << endl;
        conn->send(buf);
    }

main函数

int main(){

    EventLoop loop; //epoll
    InetAddress addr("127.0.0.1",6000);
    ChatServer server(&loop,addr,"ChatServer");
    server.start();         
    loop.loop();        //epoll_wait以阻塞方式等待新用户连接或读写事件等
    return 0;

}

全部代码:

/*muduo网络库给用户提供了两个主要的类
TcpServer:用于编写服务器程序的类
TcpClient:用于编写客户端程序的类

将epoll和线程池封装起来,好处是能够把网络的I/O代码与业务代码区分开
*/

#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;


/*基于muduo网络库开发服务器程序
1.组合TcpServer对象
2.创建EventLoop事件循环对象的指针
3.明确TcpServer构造函数需要什么参数,输出ChatServer的构造函数
4.在当前服务器类的构造函数中,注册处理连接和读写事件的回调函数
5.设置合适的服务端线程数量,muduo会自动划分I/O线程和worker线程
*/
class ChatServer
{
public:
    ChatServer(EventLoop* loop,                 //事件循环——Reactor反应堆
            const InetAddress& listenAddr,      //IP和端口
            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:
    //专门处理用户连接创建和断开    epoll
    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();
        }
    }
    //专门处理用户读写事件    
    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;
    EventLoop *_loop;       //epoll
};


int main(){

    EventLoop loop; //epoll
    InetAddress addr("127.0.0.1",6000);
    ChatServer server(&loop,addr,"ChatServer");
    server.start();         
    loop.loop();        //epoll_wait以阻塞方式等待新用户连接或读写事件等
    return 0;

}

3.3 服务器程序执行

3.3.1 g++编译

g++ -o server muduo_server.cpp -lmuduo_net -lmuduo_base -lpthread

终端输入上述语句,其中,g++ -I头文件搜索路径 -L库文件搜素路径 -l库名称,执行server文件,结果如下:

可以看到客户端登录成功后,信息成功回显!

telnet 127.0.0.1 6000

3.3.2 CMake编译

可以查看Linux环境下是否有CMake,有muduo库其实就已经有CMake了,通过下面的命令查看版本号

cmake -version

在3.2文件的同级目录下,创建CMakeLists.txt文件,分别写出编译选项、需要编译的源文件列表、可执行文件存储的路径、生成可执行文件、以及链接的库文件

cmake_minimum_required(VERSION 3.0)
project(main)

# 配置编译选项
set(CMAKE_CXX_FLAGS ${CMAKE_CXX_FLAGS} -g)  # 可调试可执行文件

#配置头文件搜索路径
# include_directories()

#配置库文件搜索路径
# link_directories()

# 设置需要编译的源文件列表
set(SRC_LIST ./muduo_server.cpp)
# 设置可执行文件存储的路径
set(EXECUTABLE_OUTPUT_PATH ${PROJECT_SOURCE_DIR}/bin)


# 把.指定路径下的所有源文件放入SRC_LIST变量名中
# aux_source_directory(.SRC_LIST)

# 表示生成可执行文件server,由SRC_LIST变量定义的源文件而来
add_executable(server ${SRC_LIST})  # 生成可执行文件

#表示server这个目标程序,需要链接muduo_net muduo_base pthread库文件
target_link_libraries(server muduo_net muduo_base pthread)

可以看到,这里将可执行文件放在了项目文件的bin文件夹下,执行server文件同样回显相同的结果

至此,muduo网络库的示例代码与实验完毕,期待后续项目的更新把~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/485583.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

mysql 用户管理-账户管理

学习了《mysql 用户管理-权限表》。接着学习更常用的的账户管理。 2&#xff0c;账户管理 MySQL提供许多语句用来管理用户账号,这些语句可以用来管理包括登录和退出MySQL服务器、创建用户、删除用户、密码管理和权限管理等内容。MySQL 数据库的安全性&#xff0c;需要通过账户管…

静态、动态代理模式(Spring学习笔记八)

代理模式是SpringAOC的底层 代理模式分为&#xff1a;静态代理模式 动态代理模式 1、静态代理 代码步骤 接口&#xff1a; package com.li.dedmo01;public interface Rent {public void rent(); }真实角色&#xff1a; package com.li.dedmo01;public class Host imple…

机器学习——贝叶斯分类器(基础理论+编程)

目录 一、理论 1、初步引入 2、做简化 3、拉普拉斯修正 二、实战 1、计算P(c) 2、计算P(x|c) 3、实战结果 1、数据集展示 2、相关信息打印 一、理论 1、初步引入 在所有相关概率都已知的理想情形下&#xff0c;贝叶斯决策论考虑如何基于这些概率和误判损失来选择最…

开源流程图表库(01):Mermaid.js生成流程图、时序图、甘特图等

一、Mermaid.js的特点 Mermaid.js是一个用于生成流程图、时序图、甘特图等各种图表的开源库。它使用简洁的文本语法来描述图表结构&#xff0c;并将其转换为可视化的图形。 Mermaid.js的主要特点包括&#xff1a; 简洁易用&#xff1a;Mermaid.js使用简单的文本语法来描述图表…

Introduction to Data Mining 数据挖掘

Why Data Mining? • The Explosive Growth of Data: from terabytes to petabytes — Data collection and data availability ◦ Automated data collection tools, database systems, Web, computerized society — Major sources of abundant data ◦ Business: Web, e-co…

RabbitMQ入门学习【附案例讲解】

文章目录 1.初识MQ1.1.同步和异步通讯1.1.1.同步通讯1.1.2.异步通讯 1.2.技术对比&#xff1a; 2、安装RabbitMQ2.1、下载镜像2.2、单机部署2.3、集群部署 3、基本结构4、RabbitMQ消息模型5、代码示例5.1、导入Demo工程5.2、publisher实现5.3、consumer实现5.4、效果 6、总结 ​…

C++项目——集群聊天服务器项目(四)MySQL数据库

今天来介绍集群聊天器项目中MySQL数据库的相关表结构 集群聊天服务器项目需要进行好友、群组的增删操作以及好友离线状态的判断&#xff0c;需要在数据库中建立相关表进行增删改查操作&#xff0c;因此项目设计了如下五个表&#xff0c;大家可以自己动动小手&#xff0c;使用S…

vulhub中Apache Shiro 1.2.4反序列化漏洞复现(CVE-2016-4437)

Apache Shiro是一款开源安全框架&#xff0c;提供身份验证、授权、密码学和会话管理。Shiro框架直观、易用&#xff0c;同时也能提供健壮的安全性。 Apache Shiro 1.2.4及以前版本中&#xff0c;加密的用户信息序列化后存储在名为remember-me的Cookie中。攻击者可以使用Shiro的…

【蓝桥杯入门记录】中断系统

一、中断概述 &#xff08;1&#xff09;处理器和外部设备的数据传输方式 无条件传输&#xff1a;处理器不比为了了解外部设备状态&#xff0c;直接进行数据传输&#xff0c;用于指示灯和蜂鸣器灯等简单设备查询方式&#xff1a;传输前&#xff0c;一方先查询另一方的状态&am…

CVPR 2024中有哪些值得关注的视频生成和视频编辑方向的论文?

Diffusion Models视频生成-博客汇总 前言:轰轰烈烈的CVPR 2024所有accept paper已经全部公开,随着Sora的爆火,视频生成和视频编辑是目前计算机视觉最火热的方向,受到了很多人的关注。这篇博客就整理盘点一下有哪些值得关注的视频生成和视频编辑方向的论文?值得做这个方向的…

Day56-LNMP架构扩展为集群模式实战精讲

Day56-LNMP架构扩展为集群模式实战精讲 1. 企业级标准部署知乎产品wecenter1.1 部署知乎软件Wecenter 2. 企业级迁移数据库到独立服务器2.1 为什么要进行数据库的拆分2.2 数据库拆分架构演变过程&#xff0c;如下图所示2.3 数据库拆分环境规划2.4 数据库拆分架构详细步骤2.4 we…

【计算机网络】启程

&#x1f4dd;本文介绍 本文为计算机网路系列的开始篇&#xff0c;会介绍一下使用的书籍和自己做的思维导图。 &#x1f44b;作者简介&#xff1a;一个正在积极探索的本科生 &#x1f4f1;联系方式&#xff1a;943641266(QQ) &#x1f6aa;Github地址&#xff1a;https://githu…

对于组件通信的深刻理解

父组件传递数据给子组件 props传递数据 父组件在子组件的标签上写自定义的属性,属性值是自己的变量,当渲染到子组件时,执行props会找自定义属性,内存了变量的内存,可访问到,写props,会生成vue实例的时候,将props的变量赋给,值找变量内存存入变量.插值语句等可访问.父组件会变…

【ArcGIS微课1000例】0106:ArcGIS制作风向频率(风速)玫瑰图

文章目录 一、效果预览二、加载数据三、创建图表四、图表修饰五、保存图片一、效果预览 在ArcGIS制作的风向频率玫瑰图最终效果如下所示: 二、加载数据 加载配套实验数据包中0106.rar中的excel数据,然后右键→打开。 三、创建图表 1. 创建图表。右击打开属性表,选择表选项…

硅谷甄选项目笔记

硅谷甄选运营平台 此次教学课程为硅谷甄选运营平台项目,包含运营平台项目模板从0到1开发&#xff0c;以及数据大屏幕、权限等业务。 此次教学课程涉及到技术栈包含***:vue3TypeScriptvue-routerpiniaelement-plusaxiosecharts***等技术栈。 一、vue3组件通信方式 通信仓库地…

在 Windows 中安装配置并启动运行 Jenkins【图文详细教程】

安装 Jenkins 的系统要求&#xff1a; 最少 256MB 可用内存最少 1GB 可用磁盘空间JDK 8 / 11 /17&#xff08;Jenkins 是用 Java 写的&#xff0c;打包成 war 包&#xff09; 查看 JDK 的版本 Java JDK 在 Windows 中安装可以参考&#xff1a;https://www.yuque.com/u27599042/…

JVM快速入门(2)HotSpot和堆、新生区、永久区、堆内存调优、JProfiler工具分析OOM原因、GC(垃圾回收)、JVM经典面试笔试题整理

5.6 HotSpot和堆 5.6.1 Hotspot 三种JVM&#xff1a; Sun公司&#xff0c;HotspotBEA&#xff0c;JRockitIBM&#xff0c;J9 VM&#xff0c;号称是世界上最快的Java虚拟机 我们一般学习的是&#xff1a;HotSpot 5.6.2 堆 Heap&#xff0c;一个JVM只有一个堆内存&#xff0c…

Py之scikit-learn-extra:scikit-learn-extra的简介、安装、案例应用之详细攻略

Py之scikit-learn-extra&#xff1a;scikit-learn-extra的简介、安装、案例应用之详细攻略 目录 scikit-learn-extra的简介 scikit-learn-extra的安装 scikit-learn-extra的案例应用 1、使用 scikit-learn-extra 中的 IsolationForest 模型进行异常检测 scikit-learn-extra…

什么是LLM Agent它如何运作?

什么是LLM Agent它如何运作&#xff1f; 代理背后的主要直觉是一个模型&#xff0c;使用大型语言模型作为其中央计算引擎来推理问题、计划解决问题并使用一组工具来解决问题。 现实世界中的实际任务没有一步解决方案。它们通常需要多个相关和独立的步骤才能完成。对于“A 公司…

AI智能视频粗剪解决方案助力企业高效运营

面对海量的视频素材&#xff0c;如何快速、准确地检索和剪辑&#xff0c;成为摆在企业面前的一大难题。美摄科技凭借其在人工智能领域的深厚积累&#xff0c;推出了领先的AI智能视频粗剪解决方案&#xff0c;为企业提供了高效、智能的视频处理新选择。 美摄科技的AI智能视频粗…
最新文章