文章目录
- 前言
- 项目简单架构图
- 1.0版本:单服务器
- 2.0版本,横向扩充,负载均衡
- 3.0版本,调优
- 代码
- 讲解
- setConnectionCallback
- setMessageCallback
前言
准备翻新一下我那个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);
//成功解耦
这一段就等业务层里面在看吧。
虽然好一段时间没有看设计模式了,不过 “松耦合,高内聚” 的思想还是深入我心的。