在之前的章节中,我们对服务端系统的设计实现原理进行了剖析,在这一章中,我们将对服务端框架进行实际运用,实现一款运行于内网环境的聊天系统。该聊天系统由客户端与服务器两部分组成,同时服务端通过数据库维护用户的账号信息。本章将重点介绍如何运用该服务端框架进行服务器业务逻辑开发。

聊天系统功能分析

  本聊天系统只作为服务端框架的运用展示,因此仅限于最基本的局域网聊天工具,数据传输均采用为明文形式,并不在安全性上进行深入探讨。具体功能需求分析如下:

  • l  用户注册:提供新用户账号注册功能,注册内容包括新用户的账号名和密码。服务器需要检测注册信息的有效性,并将注册用户的信息保存在数据库中。
  • l  用户登录:用户根据账号密码进行登录操作,若未进行登录操作将无权限请求聊天相关消息。服务器查询数据库验证用户账号信息,同时判断该用户是否已经登陆。用户账号不允许重复登录。
  • l  在线好友更新:已登录用户可以主动向服务器查询当前在线的好友信息。服务器同样应该及时向所有登录用户推送好友上线及下线信息。
  • l  好友聊天:已登录用户可以向在线状态的任意好友发送聊天消息,同时也能接收到来自其他在线状态的好友发送过来的消息。
  • l  群体聊天:已登录用户可以同时向所有在线状态的好友发送群体聊天消息。同时也能接收到来自其他在线状态的好友发送的群体聊天消息。

聊天系统服务端实现

数据库实现

  聊天系统需要维护用户的账号信息,记录注册的新账号,并对每次登录的用户信息进行校验。我们选用轻量级的Mysql作为数据库管理系统,并选择Mysql++作为Mysql的API操作库。

  整个聊天系统只建立了一个名为UserInfo表结构,如表5-1所示。同时根据Mysql++封装了两个操作接口,分别添加新用户账号信息,以及根据账号名和密码查询该用户是否存在。

表5-1 用户账号信息表UserInfo

字段名

数据类型

关键字

约束

含义

id

int(20)

Y

unique

唯一标识符

username

varchar(20)

N

unique,not null

账号名

Password

varchar(20)

N

not null

密码

  由于数据库操作涉及网络传输及磁盘处理,因此属于耗时操作。而在服务端框架Reactor反应池中的所有处理必须在非阻塞环境下进行,如果进行耗时操作将会阻塞当前线程,导致其它消息事件得不到及时处理。因此在Reactor中进行数据库操作应该以异步的方式进行,可以通过创建工作线程池的方式处理数据库等耗时操作。但是在本聊天系统中,只有在注册新用户和用户登录两个场景中才涉及数据库操作。为了简化开发模型,我们在此直接调用数据库相关的操作。

  数据库部分并非本文论述重点,因此不再做进一步介绍。

聊天设备类型及消息事件实现

  通过分析功能需求,可以得知客户端连接主要存在两种状态下的消息请求,分别为临时状态和登录状态。在临时状态下需要能够请求注册新用户和登录操作,在登录状态下需要能够请求当前在线用户信息,向某个用户发送消息类型和广播消息类型。并且由于服务器存在登录超时的机制,客户端连接不管在临时状态还是登录状态,均需定时向服务器发送心跳消息。

  客户端的两种状态正好对应于服务端框架制定的两种设备类型,即临时设备类型和登录设备类型。如图5-1所示。我们为临时设备添加新的注册账号消息事件,用于向所有与服务端建立了连接的客户进行注册操作。同时我们创建一个新的设备类型,名为ChatType设备类型,并且继承于登录设备类型,用于专门处理登录后和聊天相关的消息事件。

图5-1 聊天设备类型及消息事件

  临时设备类型新添加的消息事件介绍如下:

  • l  registerUsr:注册新的用户账号。客户端在请求该消息事件时需同时传入期待注册的账号名和密码。服务器接收到该消息事件后,将校验账号名和密码是否合法,然后调用数据库新账户注册接口,进行账户注册。最后根据数据库实际插入结果向客户端发送注册结果信息。由于该消息事件无需请求权限,任何连接该服务器的客户均可进行注册操作。为了防止存在某些客户进行恶意注册现象,在实际实现中添加了邀请码信息。服务端将会验证客户端传来的邀请码是否正确,如果邀请码正确才会实际进行注册操作。
  • l  heartMsg:心跳消息。客户端将会定期发送心跳消息。服务器接收到该消息事件后,将会在超时队列更新该连接的超时信息。在事件回调实现中,并不会对该消息进行进一步业务处理,而是直接返回。

  ChatType为新创建的继承于默认登录设备的设备类型,具体实现的消息事件介绍如下:

  • l  askFriends:请求当前在线的用户信息。客户端会在登录成功后,向服务器请求该消息事件。服务器收到该消息事件后,将会查询除请求者外,当前其他所有在线的ChatType类型的用户信息,并整理发送给客户端。该数据用于客户端对在线用户列表的初始化工作。
  • l  chatSend:向一个指定在线用户发送聊天消息。客户端将会把聊天消息的内容,自己的信息,及期望接收该聊天消息的用户信息发送给服务器。服务端接收到该消息事件后,将会尝试找寻该接收用户,并将聊天消息和发送者消息转发给该接收用户,并向发送用户返回发送成功消息事件。如果该接收用户并不在线,将只会向发送用户返回发送失败的消息事件。
  • l  groupSend:向全体在线用户发送聊天消息。客户端将会把聊天消息及自身信息发送给服务器。服务器接收到该消息后,将会遍历除请求者外,当前其他所有在线的ChatType类型的用户,并将该聊天消息和发送者信息转发给所有遍历的用户,并向发送用户返回发送成功的消息。
  • l  heartMsg:心跳消息。同临时设备新添加的heartMsg的心跳消息。

聊天设备的生命周期实现

  虽然我们已经为聊天客户端的连接制定了具体的设备类型和消息事件,但是一些和连接状态相关的业务逻辑仍然需要我们设计。比如如何制定与聊天客户端相关的具体登录过程?当某个聊天客户端已经成功登录后,怎样第一时间通知其他客户端该账号已上线?当某个聊天客户端退出时,又如何通知其他客户端该账号已下线?这些与客户端状态相关的操作都依赖上一章节制定的连接生命周期机制来进行管理。

图5-2 聊天设备的生命周期

  具体的聊天设备的连接生命周期实现如图所示。我们只需对具体业务相关的生命周期接口的实现进行重写即可,因此无需进行更改的生命周期接口并未被列出。需要被重写的接口实现介绍如下:

  • l  onLoginCheckMsg():当客户端执行登录操作,且选择登录类型为ChatType设备类型时,将会执行该接口。此时系统将会检查客户端传来的账号名和密码是否合法,如果合法将会调用数据库的账号密码查询接口,判断该账户是否存在。如果该账户存在,则返回True,并将控制权交还给系统进行进一步的登录操作;如果该账户不存在,则向该客户端发送不存在该账号信息,此次登录失败的消息,并直接返回False退出整个登录操作。
  • l  onLoginSuccessMsg():当登录类型为ChatType的客户端进行登录操作,执行onLoginCheckMsg()返回True,且系统整个登录过程成功时,将会执行该接口。此时系统将会进行两件工作:首先向该客户端发送登录成功的消息;其次将会遍历除该用户本身外的其他已登录的ChatType类型的用户,并向这些用户发送通知消息表明该新用户已上线。这样每个登录用户只需在第一次登录的时候向服务器请求当前所有在线用户信息,而新的用户上线或老用户下线等信息将由服务器主动推送,而无需客户端定期请求维护。
  • l  onLoginFailureMsg():当登录类型为ChatType的客户端进行登录操作,执行onLoginCheckMsg()返回True,但整个登录过程失败时,将会执行该接口。一般该接口被执行将表明此用户账号已经被登录,服务器无法对相同账号进行重复登录操作。此时服务器将会向客户端发送重复登录,此次登录失败的消息,并退出整个登录操作。
  • l  onLogoutMsg():当登录类型为ChatType的客户端进行注销操作时,将会执行该接口。服务器将向该客户端返回注销成功的消息,并对该连接进行进一步的注销操作。
  • l  releaseConnNode():当登录类型为ChatType的客户端进行注销成功,或者连接由于超时或者其它异常原因被迫退出时,将会执行该接口。此时服务器将会遍历除该用户本身外的其他已登录的ChatType类型的用户,并向这些用户发送通知消息表明该用户已经下线。同时系统将会把该用户连接所申请的相关资源进行释放工作。
  • l  onOverTime():当服务器一段时间内未收到该客户端有效消息或心跳消息,将会被判超时,并执行该接口。一般执行该接口时表明该客户端已经由于异常而停止工作,但是并未主动断开连接。此时服务器将尝试向该客户端发送一条该连接已经超时的消息。由于客户端已经处于异常状态,并不能保证该条消息能够被客户端接收。然后服务器将会执行releaseConnNode()接口,并强制断开该客户端连接。

聊天系统展示

  该聊天系统部署于局域网内,并可通过C#实现的客户端进行相关操作。运行场景截图如下:

图5-3 登录及注册窗口

  如图5-3为客户端的登录窗口与注册窗口。注册窗口通过登录窗口的注册按键打开,并可进行新账号的注册操作。同时登录窗口可以进行账号的登录操作,如果登录失败将会显示具体失败原因;如果登录成功,将会进入客户端主界面。

图5-4 主界面及好友聊天窗口

  如图5-4为客户端的主界面及聊天窗口。在主界面中显示了当前登录的用户名及当前在线的用户名列及数量。可以单击具体用户名进入与该用户的聊天窗口。在聊天窗口中,可以在输入栏输入消息,并点击发送键或按下回车键发送该消息。如果收到其他用户的聊天消息,客户端也会主动弹出该用户的聊天窗口,并显示接收的的聊天信息。在主界面的下方还存在群聊按键,点击可以进入群聊窗口。

图5-5 群聊窗口

  如图5-5为群聊窗口。在群聊窗口中可以接收到打开了该窗口的其他用户发送的聊天信息。同时自己也能输入相关聊天消息并发送给所有该窗口下的其他用户。

转载于:https://www.cnblogs.com/moyangvip/p/5586045.html

C++服务器设计(七):聊天系统服务端实现相关推荐

  1. java毕业生设计信息工程学院办公经费管理系统服务端计算机源码+系统+mysql+调试部署+lw

    java毕业生设计信息工程学院办公经费管理系统服务端计算机源码+系统+mysql+调试部署+lw java毕业生设计信息工程学院办公经费管理系统服务端计算机源码+系统+mysql+调试部署+lw 本源 ...

  2. 基于阿里云用C/C++做了一个http协议与TCP协议的web聊天室的服务器——《干饭聊天室》

    基于阿里云用C/C++做了一个http协议与TCP协议的web聊天室的服务器--<干饭聊天室> 在这里首先感谢前端小伙伴飞鸟 前端技术请看一款基于React.C++,使用TCP/HTTP协 ...

  3. 盛大游戏李阳:龙之谷服务器设计

    原文链接:http://www.gameres.com/779278.html 9月23日,首届"梦想·匠心"腾讯游戏开发者大会于深圳举行,在技术分论坛上,盛大游戏<龙之谷& ...

  4. 并发型服务器响应方式,Java NIO 在并发型服务器设计中的应用.pdf

    Java NIO 在并发型服务器设计中的应用 丁辉 北京邮电大学PCN&CAD 中心,北京(100876 ) E-mail :lvsehaier@ 摘 要:本文分析了应用传统阻塞型网络I/O ...

  5. java计算机毕业设计高校多媒体设备运维管理系统服务端MyBatis+系统+LW文档+源码+调试部署

    java计算机毕业设计高校多媒体设备运维管理系统服务端MyBatis+系统+LW文档+源码+调试部署 java计算机毕业设计高校多媒体设备运维管理系统服务端MyBatis+系统+LW文档+源码+调试部 ...

  6. C#下如何实现服务器+客户端的聊天程序

    C#下如何实现服务器+客户端的聊天程序 最近也在接触SOCKET编程,在当今这样一个网络时代,很多技术都以网络为中心在诞生,至少我认为是这样的,而SOCKET套接字接口,在实现网络通讯上处于关键地位, ...

  7. [七]RabbitMQ-客户端源码之AMQPImpl+Method

    欢迎支持笔者新作:<深入理解Kafka:核心设计与实践原理>和<RabbitMQ实战指南>,同时欢迎关注笔者的微信公众号:朱小厮的博客. 欢迎跳转到本文的原文链接:https: ...

  8. java仿聊天室项目总结_Java团队课程设计-socket聊天室(个人总结)

    Java团队课程设计-socket聊天室(个人总结) 一.团队课程设计博客链接 二.本人负责模块或任务说明 任务1 服务端对socket线程的接受以及对客户端的数据转发操作 任务2 数据库的查找,添加 ...

  9. 并发服务器设计思路,参考apache学习UDP和QoS,研究成果

    研究了快1个月的服务器架构,把研究成果记录一下. 参考的有:Apache  vlc  ACE  ftp 我主要需要其中的并发处理,内存管理,TCP/UDP.QoS,速度限制等方面的内容,所以着重说这几 ...

最新文章

  1. 【C 语言】文件操作 ( 文件结尾判定 )
  2. 基于消息队列 RocketMQ 的大型分布式应用上云较佳实践
  3. 云原生高可用技术体系的构建
  4. 大河抽奖盲盒运营版 1.9.12开源版
  5. 1gb 云服务器 和2gb的区别_(12)虚拟主机/VPS/云主机/服务器有什么区别?
  6. web项目开发最佳做法
  7. centos6.5-64安装zabbix2.4
  8. KeyMob手机聚合平台已集成多家移动广告平台
  9. 在Fedora 28上安装LimeSurvey CE
  10. html点击复制一段文字内容,js实现点击按钮复制文本功能
  11. Sublime Text 3破解补丁
  12. 完全教程 Aircrack-ng破解WEP、WPA-PSK加密利器[zz]
  13. 微商行业面临洗礼,微盟萌店能否完成“神补刀”?
  14. 01 JavaScript的前世今生
  15. FFmpeg音视频播放器系列(第三篇:seek实现播放进度控制)
  16. 中国象棋游戏Chess(1) - 棋盘绘制以及棋子的绘制
  17. mac 取消 “Ctrl+空格“切换输入法
  18. java去掉字符串的空格_如何去掉字符串中的空格?
  19. 德丰杰全球创始人兼董事长Tim Draper让人吃惊的投资
  20. 程序员笔记本性能需求分析--2020年--极致性价比(真香定律)

热门文章

  1. CellSet 遍历
  2. 在wp中,使用NavigationService.Navigate导航页面出现错误
  3. 程序员编程艺术第十一章:最长公共子序列(LCS)问题
  4. WMI 使用教程_.NET 入门教程
  5. CSS3实现多页签图片缩放切换效果
  6. 事件传递 java_将事件传递/发送到父组件?
  7. c语言科学计数法_C入门:C语言中数据的储存(上)
  8. webpackjsonp 还原_具有催化CO2还原性能的非贵金属配合物的配体设计
  9. com技术内幕 代码_CFan科学院:零门槛极速抠图技术探秘
  10. linux 用户 组区别吗,Linux系统的用户和用户组有什么区别