【项目】微服务及时通讯系统:编写核心类

2024-08-24 09:44

本文主要是介绍【项目】微服务及时通讯系统:编写核心类,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 前言
  • 1. 核心数据结构
    • 1.1 用户信息
    • 1.2 会话信息
    • 1.3 消息信息
  • 2. 建立目录
  • 3. 编写代码
    • 3.1 用户信息
    • 3.2 会话信息
    • 3.3 消息信息
    • 3.4 工具函数
  • 4. data.h 完整代码
  • 总结

前言

在构建现代微服务架构的即时通讯系统时,核心数据结构的设计是至关重要的。它们不仅决定了系统的性能和可扩展性,而且也影响着用户交互的直观性和便捷性。本文将深入探讨即时通讯系统中的三个核心数据结构:用户信息、会话信息和消息信息,以及它们是如何在C++和Qt框架下实现的。通过详细解析这些数据结构的设计和实现,我们希望能够为开发者提供一个清晰的指导,帮助他们构建高效、稳定且用户友好的即时通讯应用。

1. 核心数据结构

1.1 用户信息

1.2 会话信息

用户和用户之间,聊天会话:
例如:

用户A:有2个好友:B、C
此时A可以和B单聊,也能和C单聊(2个聊天会话)
此时这个群组,也对应一个聊天会话

聊天程序中,会话的生命周期,是比较的;一直持续到把对方好友删除/退出群组才会随之销毁。

1.3 消息信息

  1. 文本消息
  2. 图片消息
  3. 文件消息
  4. 语音消息

2. 建立目录

在文件中显示
在这里插入图片描述
创建一个文件夹存放目录
在这里插入图片描述
新建一个文件
在这里插入图片描述
添加进来
在这里插入图片描述
在这里插入图片描述
此时cmake 就自动添加上了
在这里插入图片描述

qt_add_executable:指定编译qt需要依赖哪些文件(源代码,肯定是要依赖的)

关于命名空间的约定(这种约定,仅限于当前项目):
如果代码所在的文件,就是在项目的顶层目录中,此时就直接使用全局命名空间(不手动指定)
在这里插入图片描述
如果代码所在的文件,在某个子目录中;此时,就指定一个和目录名字相同的命名空间。
在这里插入图片描述

使代码中的命名空间的结构和文件在目录中结构一致。(尤其是在目录结构更复杂,嵌套多层)

3. 编写代码

3.1 用户信息


/// 用户信息
class UserInfo {
public:QString userId = "";         // 用户编号QString nickname = "";       // 用户昵称QString description = "";    // 用户签名QString phone = "";          // 手机号码QIcon avatar;                // 用户头像
};
QString userId;

使用字符串的方式来作为id,可以有更灵活的方式来生成。(也为了能够适应分布式后端)

mysql 数据库,支持自增主键:
如果是单个节点的 mysql,用上述方式没有任何问题
但是,如果是多个节点的分布式
mysql,就无法使用整数自增组件了 分布式 mysql 下,很可能需要针对用户信息“分库分表”

后续中可以通过例如:uuid 这样的方式,或者是 雪花算法这样的方式 来生成分布式系统中唯一id

3.2 会话信息


/// 会话信息
class ChatSessionInfo {QString chatSessionId = "";      // 会话编号QString chatSessionName = "";    // 会话名字,如果是会话是单聊,名字就是对方的昵称;如果是群聊,名字就是群聊的名称Message lastMessage;             // 表示最新的消息QIcon avatar;                    // 会话头像,如果会话是单聊,头像就是对方的头像;如果是群聊,头像群聊的头像QString userId = "";             // 对于单聊来说,表示对方的用户 id,对于群聊设置为 ""
};
ChatSessionInfo

前面谈到的“会话”都是针对 聊天过程中的,组织消息的 会话。
后面还会涉及到,客户端连接到服务器之后,也有一个“登录”用到的会话

在这里插入图片描述

Message lastMessage;

这个内容就是为了在会话列表中,能够起到“显示-提示”这样的效果。
在这里插入图片描述
针对会话信息来说:

一个会话,里面其实是可以包含多个用户的
这个会话里具体有哪些用户,
后续会通过单独的方式进行组织管理。

此处列出 QString userId;表示的含义是:

  1. 如果会话是单聊会话,此时 userId 表示“对方”的用户 id
  2. 如果会话是群聊会话,此时 userId 设为 “”,后续通过其他的方式来吧完整的用户id列表拿到

会话 - 消息 是 “一对多” 这样的关系。
消息:会话id。

 QByteArray content;         // 消息发送的正文内容

如果是 文本消息,正文就是一个字符串
如果是 图片,文件,语音消息,正文就是一个“二进制序列”

| 在C/C++ 中没有 byte 这样的类型,表示 byte 都是拿 char / unsigned char 凑合一下

| char / unsigned char
正常来说,一个char(字符) 不一定是一个字节

  • 对于 应用 ascii 来说,一个字符就是一个字节
  • 对于 中文 gbk 编码来说,一个字符就是2个字节
  • 对于 中文 utf8 编码来说,一个字符就是3个字节

由于 C/C++ 有点太老了,对于这一块的支持,比较有限
C++ 中,一个 char 就是固定的一个字节了。

| 比方说,C++中:std::string name = "张三";
C++ 中,通过代码取出“三”这个汉字,老麻烦了!

  1. 先判定是那种编码方式
  2. 计算第二个汉字 所属的范围
  3. 取字符串子串

| 像其他主流语言,Java/Python 之类的
string s = "张三";
s.charAt(1) => "三"

| 相比之下,Qt做了更好的处理;QString 就对上述情况处理的更好了。 Qt 也明确区分了 “字节” 和 “字符”

QString fileId;             // 文件的身份标识,当类型为 文件,图片,语言 的时候,才有效;当消息为文本,则为""

文件/图片/语言 这些消息,体积可能是比较大的!(网络带宽)

  • 一旦一个聊天会话中,包含多个上述这样的消息,就会使从服务器消息列表这样的操作,变得非常低效。
    一般的做法,都是“获取消息列表”,只是拿到文件/图片/语言 消息的 filed 等到客户端得到“消息列表”之后,再更具拿到的filed,给服务器发送额外的请求,获取文件内容。(化整为零)
QString fileName;           // 文件名称,只是当消息类型为 文件消息时, 才有效,其他消息均为 ""

虽然图片/语音,这两个也是“文件”但是文件名不需要显示到界面上。
对于文件消息,希望界面上显示“文件名”,点击之后可以进行“另存为”这样的操作。

3.3 消息信息


/// 消息信息
enum MessageType {TEXT_TYPE,      // 文本消息IMAGE_TYPE,     // 图片消息FILE_TYPE,      // 文件消息SPEECH_TYPE,    // 语音消息
};class Message {
public:QString messageId = "";          // 消息的编号QString chatSessionId = "";      // 消息所属会话的编号QString time = "";               // 消息时间,通过“格式化”时间的方式来表示, 形如:06-07 12:00:00MessageType messageType = TEXT_TYPE; // 消息类型UserInfo sender;                 // 发送者的信息QByteArray content = "";         // 消息发送的正文内容QString fileId = "";             // 文件的身份标识,当类型为 文件,图片,语言 的时候,才有效;当消息为文本,则为""QString fileName = "";           // 文件名称,只是当消息类型为 文件消息时, 才有效,其他消息均为 ""// 此处 extraInfo 目前只是再消息类型为文件消息时,作为“文件名”补充。static Message makeMessage(MessageType messageType, const QString& chatSessionId, const UserInfo& sender, const QByteArray& content, const QString& extraInfo){if (messageType == TEXT_TYPE) {return makeTextMessage(chatSessionId, sender, content);} else if (messageType == IMAGE_TYPE) {return makeImageMessage(chatSessionId, sender, content);} else if (messageType == FILE_TYPE) {return makeFileMessage(chatSessionId, sender, content, extraInfo);} else if (messageType == SPEECH_TYPE) {return makeSpeechMessage(chatSessionId, sender, content);} else {// 触发了未知消息类型return Message();}}private:// 通过这个方法生成一个唯一的 messageIdstatic QString makeId() {return "M" + QUuid::createUuid().toString().sliced(25, 12);}static Message makeTextMessage(const QString& chatSessionId, const UserInfo& sender, const QByteArray& content) {Message message;// 此处需要确保,设置的 messageId 是 “唯一” 的message.messageId = makeId();message.chatSessionId = chatSessionId;message.sender = sender;message.time = formatTime(getTime()); // 生成一个格式化时间message.content = content;message.messageType = TEXT_TYPE;// 对于文本消息来说,这两个属性不使用,设为""message.fileId = "";message.fileName = "";return message;}static Message makeImageMessage(const QString& chatSessionId, const UserInfo& sender, const QByteArray& content) {Message message;// 此处确保,设置的 messageId 是“唯一”的message.messageId = makeId();message.chatSessionId = chatSessionId;message.sender = sender;message.time = formatTime(getTime());message.content = content;message.messageType = IMAGE_TYPE;// fileId 后续使用的时候进一步设置message.fileId = "";// fileName 不使用,直接设为“”message.fileName = "";return message;}static Message makeFileMessage(const QString& chatSessionId, const UserInfo& sender, const QByteArray& content, const QString& fileName) {Message message;message.messageId = makeId();message.chatSessionId = chatSessionId;message.sender = sender;message.time = formatTime(getTime());message.content = content;message.messageType = FILE_TYPE;// fileId 后续使用的时候进一步设置message.fileId = "";message.fileName = fileName;return message;}static Message makeSpeechMessage(const QString& chatSessionId, const UserInfo& sender, const QByteArray& content) {Message message;message.messageId = makeId();message.chatSessionId = chatSessionId;message.sender = sender;message.time = formatTime(getTime());message.content = content;message.messageType = SPEECH_TYPE;// fileId 后续使用的时候进一步设置message.fileId = "";// fileName 不使用,直接设为“”message.fileName = "";return message;}
};
  • 工厂方法:(工厂模式)
    解决 C++/Java 等语言中,构造函数,不够用的问题。

虽然平时构建对象,都是通过对 构造函数 来完成。
如果有不同的方式来构造对象,此时构造函数就不太够用了。

比方说 “点”

class Point {
public:Point(double x, double y);  // 直角坐标的方式构造Point(double r, double a);	// 极坐标的构造方式
}

够高函数想要提供不同版本,必须要 “重载” 要求 函数名 相同(都是一样)
参数的个数/类型不同

使用普通的函数来实现不同的构造方式;普通函数,函数名可以随便起,也就不再受到“重载”的约束了。

这里说“普通”,一般都要使用 static 修饰的 静态函数。(调用已有的构造函数,根据不同的需求,进一步的实现初始化的细节)

相比于 UserInfoChatSessionInfo, Message 是更需要“工厂模式”的,Message 需要支持多种构造方式;

  1. 文本消息
  2. 图片消息
  3. 语音消息
  4. 文件消息

messageId 是一个“唯一”这样的内容
UUID 这个东西,背后是一套算法,通过这个算法,就能生成“全球唯一的身份标识”,Qt对这个算法也是有封装的。
在这里插入图片描述
这一串,其实是16进制的整数,实际开发中,为了提高“可读性”也可以截取uuid中一部分来进行使用。

file.write(content);
file.flush();
file.close();

flush: 刷新缓冲区

3.4 工具函数


/// 工具函数,后续很多模块可能都要用到
static inline QString getFileName(const QString& path) {QFileInfo fileInfo(path);return fileInfo.fileName();
}// 封装一个“宏”作为打印日志的方式
#define TAG QString("[%1:%2]").arg(model::getFileName(__FILE__), QString::number(__LINE__))// qDebug 打印字符串的时候,就会自动加上" "
#define LOG() qDebug().noquote() << TAG// 要求函数的定义如果写在 .h 中,必须加 static 或者 inline(当然两个都加也可以),避链接阶段出现“函数重定义”的问题
static inline QString formatTime(int64_t timestamp) { // 为了防止在2038年溢出,用64位整数// 先把时间戳,转换成QDateTime 对象QDateTime dateTime =  QDateTime::fromSecsSinceEpoch(timestamp);// 把 QDateTime 对象转换成“格式时间”return dateTime.toString("MM-dd HH:mm:ss");
}// 通过这个函数得到 秒级 的时间
static inline int64_t getTime() {return QDateTime::currentMSecsSinceEpoch();
}// 根据 QByteArray, 转成 QIcon
static inline QIcon makeIcon(const QByteArray& byteArray) {QPixmap pixmap;pixmap.loadFromData(byteArray);QIcon icon(pixmap);return icon;
}// 读写文件操作
// 从读取文件中,读取所有的二进制内容,得到一个 QByteArray
static inline QByteArray loadFileToByteArray(const QString& path) {QFile file(path);bool ok = file.open(QFile::ReadOnly);if (!ok) {qDebug() << "文件打开失败";return QByteArray();}QByteArray content = file.readAll();file.close();return content;
}// 把 QByteArray 中的内容,写入到某个指定的文件夹里
static inline void writeByteArryToFile(const QString& path, const QByteArray& content) {QFile file(path);bool ok = file.open(QFile::WriteOnly);if (!ok) {qDebug() << "文件打开失败";return;}file.write(content);file.flush();   // 刷新缓冲区file.close();
}

4. data.h 完整代码

#pragma once
#include <QString>
#include <QIcon>
#include <QUuid>
#include <QDateTime>
#include <QFile>
#include <QFileInfo>
#include <QDebug>// 创建命名空间
namespace model {
/// 工具函数,后续很多模块可能都要用到
static inline QString getFileName(const QString& path) {QFileInfo fileInfo(path);return fileInfo.fileName();
}// 封装一个“宏”作为打印日志的方式
#define TAG QString("[%1:%2]").arg(model::getFileName(__FILE__), QString::number(__LINE__))// qDebug 打印字符串的时候,就会自动加上" "
#define LOG() qDebug().noquote() << TAG// 要求函数的定义如果写在 .h 中,必须加 static 或者 inline(当然两个都加也可以),避链接阶段出现“函数重定义”的问题
static inline QString formatTime(int64_t timestamp) { // 为了防止在2038年溢出,用64位整数// 先把时间戳,转换成QDateTime 对象QDateTime dateTime =  QDateTime::fromSecsSinceEpoch(timestamp);// 把 QDateTime 对象转换成“格式时间”return dateTime.toString("MM-dd HH:mm:ss");
}// 通过这个函数得到 秒级 的时间
static inline int64_t getTime() {return QDateTime::currentMSecsSinceEpoch();
}// 根据 QByteArray, 转成 QIcon
static inline QIcon makeIcon(const QByteArray& byteArray) {QPixmap pixmap;pixmap.loadFromData(byteArray);QIcon icon(pixmap);return icon;
}// 读写文件操作
// 从读取文件中,读取所有的二进制内容,得到一个 QByteArray
static inline QByteArray loadFileToByteArray(const QString& path) {QFile file(path);bool ok = file.open(QFile::ReadOnly);if (!ok) {qDebug() << "文件打开失败";return QByteArray();}QByteArray content = file.readAll();file.close();return content;
}// 把 QByteArray 中的内容,写入到某个指定的文件夹里
static inline void writeByteArryToFile(const QString& path, const QByteArray& content) {QFile file(path);bool ok = file.open(QFile::WriteOnly);if (!ok) {qDebug() << "文件打开失败";return;}file.write(content);file.flush();   // 刷新缓冲区file.close();
}
/// 用户信息
class UserInfo {
public:QString userId = "";         // 用户编号QString nickname = "";       // 用户昵称QString description = "";    // 用户签名QString phone = "";          // 手机号码QIcon avatar;                // 用户头像
};
/// 消息信息
enum MessageType {TEXT_TYPE,      // 文本消息IMAGE_TYPE,     // 图片消息FILE_TYPE,      // 文件消息SPEECH_TYPE,    // 语音消息
};class Message {
public:QString messageId = "";          // 消息的编号QString chatSessionId = "";      // 消息所属会话的编号QString time = "";               // 消息时间,通过“格式化”时间的方式来表示, 形如:06-07 12:00:00MessageType messageType = TEXT_TYPE; // 消息类型UserInfo sender;                 // 发送者的信息QByteArray content = "";         // 消息发送的正文内容QString fileId = "";             // 文件的身份标识,当类型为 文件,图片,语言 的时候,才有效;当消息为文本,则为""QString fileName = "";           // 文件名称,只是当消息类型为 文件消息时, 才有效,其他消息均为 ""// 此处 extraInfo 目前只是再消息类型为文件消息时,作为“文件名”补充。static Message makeMessage(MessageType messageType, const QString& chatSessionId, const UserInfo& sender, const QByteArray& content, const QString& extraInfo){if (messageType == TEXT_TYPE) {return makeTextMessage(chatSessionId, sender, content);} else if (messageType == IMAGE_TYPE) {return makeImageMessage(chatSessionId, sender, content);} else if (messageType == FILE_TYPE) {return makeFileMessage(chatSessionId, sender, content, extraInfo);} else if (messageType == SPEECH_TYPE) {return makeSpeechMessage(chatSessionId, sender, content);} else {// 触发了未知消息类型return Message();}}private:// 通过这个方法生成一个唯一的 messageIdstatic QString makeId() {return "M" + QUuid::createUuid().toString().sliced(25, 12);}static Message makeTextMessage(const QString& chatSessionId, const UserInfo& sender, const QByteArray& content) {Message message;// 此处需要确保,设置的 messageId 是 “唯一” 的message.messageId = makeId();message.chatSessionId = chatSessionId;message.sender = sender;message.time = formatTime(getTime()); // 生成一个格式化时间message.content = content;message.messageType = TEXT_TYPE;// 对于文本消息来说,这两个属性不使用,设为""message.fileId = "";message.fileName = "";return message;}static Message makeImageMessage(const QString& chatSessionId, const UserInfo& sender, const QByteArray& content) {Message message;// 此处确保,设置的 messageId 是“唯一”的message.messageId = makeId();message.chatSessionId = chatSessionId;message.sender = sender;message.time = formatTime(getTime());message.content = content;message.messageType = IMAGE_TYPE;// fileId 后续使用的时候进一步设置message.fileId = "";// fileName 不使用,直接设为“”message.fileName = "";return message;}static Message makeFileMessage(const QString& chatSessionId, const UserInfo& sender, const QByteArray& content, const QString& fileName) {Message message;message.messageId = makeId();message.chatSessionId = chatSessionId;message.sender = sender;message.time = formatTime(getTime());message.content = content;message.messageType = FILE_TYPE;// fileId 后续使用的时候进一步设置message.fileId = "";message.fileName = fileName;return message;}static Message makeSpeechMessage(const QString& chatSessionId, const UserInfo& sender, const QByteArray& content) {Message message;message.messageId = makeId();message.chatSessionId = chatSessionId;message.sender = sender;message.time = formatTime(getTime());message.content = content;message.messageType = SPEECH_TYPE;// fileId 后续使用的时候进一步设置message.fileId = "";// fileName 不使用,直接设为“”message.fileName = "";return message;}
};
/// 会话信息
class ChatSessionInfo {QString chatSessionId = "";      // 会话编号QString chatSessionName = "";    // 会话名字,如果是会话是单聊,名字就是对方的昵称;如果是群聊,名字就是群聊的名称Message lastMessage;             // 表示最新的消息QIcon avatar;                    // 会话头像,如果会话是单聊,头像就是对方的头像;如果是群聊,头像群聊的头像QString userId = "";             // 对于单聊来说,表示对方的用户 id,对于群聊设置为 ""
};} // end model

总结

本文详细介绍了微服务即时通讯系统中的核心数据结构,包括用户信息、会话信息和消息信息,以及它们在C++和Qt环境下的具体实现。我们首先对每个数据结构的功能和属性进行了概述,然后通过代码示例展示了如何定义这些结构和相关的工具函数。特别地,我们使用了工厂模式来简化消息对象的创建过程,以支持不同类型的消息构造。此外,文中还探讨了UUID生成机制,确保了消息ID的唯一性,这对于分布式系统尤为重要。

通过本文的阅读,开发者应该能够理解并实现一个高效的消息处理系统,它不仅能够处理文本消息,还能够处理图片、文件和语音等多种类型的消息。此外,通过合理组织代码和使用适当的设计模式,可以提高代码的可维护性和可扩展性,为未来的功能扩展打下坚实的基础。最后,本文提供的代码示例和设计思路可以作为构建即时通讯系统的一个参考起点,帮助开发者快速进入项目开发阶段。

这篇关于【项目】微服务及时通讯系统:编写核心类的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1102127

相关文章

javafx 如何将项目打包为 Windows 的可执行文件exe

《javafx如何将项目打包为Windows的可执行文件exe》文章介绍了三种将JavaFX项目打包为.exe文件的方法:方法1使用jpackage(适用于JDK14及以上版本),方法2使用La... 目录方法 1:使用 jpackage(适用于 JDK 14 及更高版本)方法 2:使用 Launch4j(

Docker集成CI/CD的项目实践

《Docker集成CI/CD的项目实践》本文主要介绍了Docker集成CI/CD的项目实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录一、引言1.1 什么是 CI/CD?1.2 docker 在 CI/CD 中的作用二、Docke

什么是cron? Linux系统下Cron定时任务使用指南

《什么是cron?Linux系统下Cron定时任务使用指南》在日常的Linux系统管理和维护中,定时执行任务是非常常见的需求,你可能需要每天执行备份任务、清理系统日志或运行特定的脚本,而不想每天... 在管理 linux 服务器的过程中,总有一些任务需要我们定期或重复执行。就比如备份任务,通常会选在服务器资

SpringBoot项目引入token设置方式

《SpringBoot项目引入token设置方式》本文详细介绍了JWT(JSONWebToken)的基本概念、结构、应用场景以及工作原理,通过动手实践,展示了如何在SpringBoot项目中实现JWT... 目录一. 先了解熟悉JWT(jsON Web Token)1. JSON Web Token是什么鬼

手把手教你idea中创建一个javaweb(webapp)项目详细图文教程

《手把手教你idea中创建一个javaweb(webapp)项目详细图文教程》:本文主要介绍如何使用IntelliJIDEA创建一个Maven项目,并配置Tomcat服务器进行运行,过程包括创建... 1.启动idea2.创建项目模板点击项目-新建项目-选择maven,显示如下页面输入项目名称,选择

Jenkins中自动化部署Spring Boot项目的全过程

《Jenkins中自动化部署SpringBoot项目的全过程》:本文主要介绍如何使用Jenkins从Git仓库拉取SpringBoot项目并进行自动化部署,通过配置Jenkins任务,实现项目的... 目录准备工作启动 Jenkins配置 Jenkins创建及配置任务源码管理构建触发器构建构建后操作构建任务

TP-LINK/水星和hasivo交换机怎么选? 三款网管交换机系统功能对比

《TP-LINK/水星和hasivo交换机怎么选?三款网管交换机系统功能对比》今天选了三款都是”8+1″的2.5G网管交换机,分别是TP-LINK水星和hasivo交换机,该怎么选呢?这些交换机功... TP-LINK、水星和hasivo这三台交换机都是”8+1″的2.5G网管交换机,我手里的China编程has

使用Java编写一个文件批量重命名工具

《使用Java编写一个文件批量重命名工具》这篇文章主要为大家详细介绍了如何使用Java编写一个文件批量重命名工具,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录背景处理1. 文件夹检查与遍历2. 批量重命名3. 输出配置代码片段完整代码背景在开发移动应用时,UI设计通常会提供不

基于Qt实现系统主题感知功能

《基于Qt实现系统主题感知功能》在现代桌面应用程序开发中,系统主题感知是一项重要的功能,它使得应用程序能够根据用户的系统主题设置(如深色模式或浅色模式)自动调整其外观,Qt作为一个跨平台的C++图形用... 目录【正文开始】一、使用效果二、系统主题感知助手类(SystemThemeHelper)三、实现细节

Nginx、Tomcat等项目部署问题以及解决流程

《Nginx、Tomcat等项目部署问题以及解决流程》本文总结了项目部署中常见的four类问题及其解决方法:Nginx未按预期显示结果、端口未开启、日志分析的重要性以及开发环境与生产环境运行结果不一致... 目录前言1. Nginx部署后未按预期显示结果1.1 查看Nginx的启动情况1.2 解决启动失败的