前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >chat集群聊天室项目 代码+讲解(一):网络模块

chat集群聊天室项目 代码+讲解(一):网络模块

作者头像
看、未来
发布2021-10-09 15:34:01
1.4K0
发布2021-10-09 15:34:01
举报
请添加图片描述
请添加图片描述

文章目录

前言

准备翻新一下我那个C++集群聊天室项目的讲解博客,那个系列刚开始讲的还比较有点耐心,后面就直接堆代码了,大家互相理解理解,那确实有点触碰到我目前的极限了,能写下来就不错了。

勉强把项目整好之后我就去看muduo了(好像还有最后一篇《TCPServer》还没放出来哈,真的是因为VScode坏了,然后又有一件不得不做的事情拖住了几天时间。)目前已经粗略的捋了一遍muduo的源码,这周就把项目和源码两个系列好好的整理出来,再背背八股文,把以前的基础找回来,准备面试了。

然后,祝我大莆田无恙吧,这次的事情是突然了点。(就不放张伟了,换妈祖像。)

github地址:chat


项目简单架构图

1.0版本:单服务器

请添加图片描述
请添加图片描述

2.0版本,横向扩充,负载均衡

请添加图片描述
请添加图片描述

3.0版本,调优

时间原因还没写。

我又想了一下,这个架构有问题,先留着吧,有兴趣的朋友可以看看哪里有问题。

请添加图片描述
请添加图片描述

代码

代码语言:javascript
复制
#pragma

#include<muduo/net/TcpServer.h>
#include<muduo/net/EventLoop.h>

#include<iostream>

using namespace std;
using namespace muduo;
using namespace muduo::net;


class ChatServer{
private:
    TcpServer _server;
    EventLoop* _loop;

public:
    ChatServer(EventLoop* loop,
            const InetAddress& listenAddr,
            const string& nameArg);
    
    void start();

private:
    void onConnection(const TcpConnectionPtr& conn);
    void onMessage(const TcpConnectionPtr& conn,Buffer* buff,Timestamp time);
};
代码语言:javascript
复制
#include "chatserver.hpp"
#include "chatservice.hpp"
#include "json.hpp"

#include <functional>
#include <string>

using json = nlohmann::json;
using namespace std;
using namespace placeholders;

ChatServer::ChatServer(EventLoop *loop,
                       const InetAddress &listenAddr,
                       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));

    //设置线程数
    _server.setThreadNum(5);
}

void ChatServer::start()
{
    _server.start();
}


void ChatServer::onMessage(const TcpConnectionPtr &conn, Buffer *buff, Timestamp time){
    string buf = buff->retrieveAllAsString();

    json js = json::parse(buf);

    //通过msgid获取业务回调,进行网络模块和任务模块之间的解耦合
    auto msgHandler = ChatService::instance()->getHandle(js["msgid"].get<int>());
    
    //回调消息绑定好的事件处理器,执行相应的业务处理
    msgHandler(conn,js,time);

    //成功解耦
}

void ChatServer::onConnection(const TcpConnectionPtr &conn){
    if(!conn->connected()){ //用户断开连接

        ChatService::instance()->clientCloseException(conn);
        conn->shutdown();
    }
}

讲解

无奈,有段时间没有画类图了,不然这里放张类图可能会好看点。讲业务层的时候是需要放类图了,那个牵扯有点多了。

setConnectionCallback

代码语言:javascript
复制
_server.setConnectionCallback(std::bind(&ChatServer::onConnection, this, _1));

这个_server是一个TcpServer对象,我们就直接去看源码吧。

代码语言:javascript
复制
void setConnectionCallback(const ConnectionCallback& cb)
{ 
	connectionCallback_ = cb; 
}

一个简单的赋值。

代码语言:javascript
复制
typedef std::function<void (const TcpConnectionPtr&)> ConnectionCallback;

ConnectionCallback 是一个函数对象,需要传入一个TcpConnection对象的指针。

代码语言:javascript
复制
typedef std::shared_ptr<TcpConnection> TcpConnectionPtr;

再多久不用说啦,说下去没完没了的了。

这行代码的意思是:用户注册一个连接回调,当收到连接状态变更消息时,一并调用此回调。 就是说,不写也可以,muduo库内部有默认的连接回调,不过效果可能没那么理想化,想定制就自己写一个连接回调。

那个_1是预留参数位,调用者为muduo网络库。


setMessageCallback

代码语言:javascript
复制
_server.setMessageCallback(std::bind(&ChatServer::onMessage, this, _1, _2, _3));

概念同上,注册一个用户自己的消息回调,参数有muduo网络库传入,不过位置要注意不要乱了。

代码语言:javascript
复制
// the data has been read to (buf, len)
typedef std::function<void (const TcpConnectionPtr&,
                            Buffer*,
                            Timestamp)> MessageCallback;

这个回调函数需要的参数就比较多了,不过也合理。

什么是回调?。。。 观察者模式。


代码语言:javascript
复制
//通过msgid获取业务回调,进行网络模块和任务模块之间的解耦合
auto msgHandler = ChatService::instance()->getHandle(js["msgid"].get<int>());
    
//回调消息绑定好的事件处理器,执行相应的业务处理
msgHandler(conn,js,time);

//成功解耦

这一段就等业务层里面在看吧。

虽然好一段时间没有看设计模式了,不过 “松耦合,高内聚” 的思想还是深入我心的。

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2021/09/12 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 文章目录
  • 前言
  • 项目简单架构图
    • 1.0版本:单服务器
      • 2.0版本,横向扩充,负载均衡
        • 3.0版本,调优
        • 代码
        • 讲解
          • setConnectionCallback
            • setMessageCallback
            相关产品与服务
            负载均衡
            负载均衡(Cloud Load Balancer,CLB)提供安全快捷的流量分发服务,访问流量经由 CLB 可以自动分配到云中的多台后端服务器上,扩展系统的服务能力并消除单点故障。负载均衡支持亿级连接和千万级并发,可轻松应对大流量访问,满足业务需求。
            领券
            问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档