群聊的时候,是所有人都收到群发消息,只是程序需要判断下,我在不在这个群里,如果我在群里我继续收到消息,如果我不在群里我其实还是收到了消息,只是程序判断我不应该看到这个消息return了。
所以,所有的群聊聊天室,其实所有人都在,只是程序根据当前用户是否在这个群里做了过滤显示。---这个很重要。(可能还有更好的方案,目前我感觉这种最简单,可快速实现所有功能),这样操作可能会导致数据浪费,不过首先这个不需要流量,其次这个app的用户群里两位数,基本确定没什么高并发的数据问题。假设有先实现功能后面再优化。
假设,现在有三个人:张三(103),李四(104),王五(105)。 目前他们都是新用户,没有一个群。
1. 如何知道有哪些群?
张三,李四,王五全部在线。
张三拉李四,王五创建群a,创建后张三是群主,创建群成功的时候,张三需要本地写一个缓存(或者数据库), 我觉得数据库比较好,因为只插入一条数据就可以了。缓存你还要鼓捣数组,先取再改再存,比较麻烦。强烈建议用sqlite, 我这里按sqlite思路走。
id creater(群主) name(群名称) head_image(群头像) members(群成员) updated(群信息更新时间)
1 103 群a group1.png 103,104,105 1965646479
注: 上面是数据库字段与对应的第一条数据记录,其中members最好是新开一张表外键关联,如果你不太了解外键关联查询,可以把所有的成员用一个大的字符串表示,中间用逗号隔开。
此时没有任何人说话,那么此时只有张三一个人知道有这个群的存在。可能你会想在创建群的时候发一个群发消息告知其他人,这样做没问题只是意义不大,因为这个群里面假设李四不在线,他一样不知道群的存在,这个操作没有太大的意义,而且下面的优化操作将取代这个思路。
然后,张三说话: 大家好。消息体数据格式如下:
{
type: "text", // 文字消息
from: "103", //张三的ip
message_type: "group", //群消息, 假设为单聊,这里为single
group:{
id: 1, // 群id
creater:103, //群主,只有群主可以修改群信息,修改群信息后updated会变成当前时间戳
name: 群a, //群名字,特别注意,这个名字可能会变,怎么优化下面提到, 和updated有关
head_image: group1.png, //群头像,特别注意,这个名字可能会变,怎么优化下面提到, 和updated有关
members: 103,104,105, //群成员,这个字段非常重要, 是控制群成员的法宝
updated:1965646479 //群信息更新时间
}
}
上面是发一条群消息的数据体,里面包含了很多重要信息。
此时,李四和王五收到消息(didReceiveGroupMessage),以李四为例,收到消息后,
message_type == "group", 说明是群消息,首先检查下,自己有没有id=1这个群,发现没有,于是根据group对象在本地数据库新建写了一条数据如下:
id creater(群主) name(群名称) head_image(群头像) members(群成员) updated(群信息更新时间)
1 103 群a group1.png 103,104,105 1965646479
王五同理。
现在假设,王五不在线,那么他离线状态立马上线肯定是不知道这个群存在的。只有在线,且其他群成员发一条消息过来,他重复上面的代码即可。同一套逻辑只是时间晚了点而已。
据我观察,微信拉群组机制和上面这个一模一样。
2. 群信息变更如何处理?
假设张三修改了群名称和群头像,那么id=1的群的updated等于修改时候的时间戳,张三再发消息的时候,李四和王五在didReceiveGroupMessage收到消息的时候
1. 检查自己是否有id=1的群, (发现有,不需要插入新群)
2. 核对updated是否相等,发现不相等(说明群主更新了群信息,更新本地这条群数据)
这样就简单实现了群信息的更新与同步。
3. 如果在群列表确定我是否有某个群,或者说我有哪些群?
这个很简单,直接读取数据库即可。一切以数据库为准。
4. 如果李四被群主张三从群里移除了怎么处理?
这个操作还是在didReceiveGroupMessage里面操作,其实上面已经做了,就是群信息变更里面。
核对updated是否相等,发现不相等(说明群主更新了群信息,更新本地这条群数据),
比如张三没改头像,没改群名称,就删了李四。这同样是更改了群信息。updated同样发生了变化。李四在收到消息后,
1. 检查自己是否有id=1的群, (发现有,不需要插入新群)
2. 核对updated是否相等,发现不相等(说明群主更新了群信息,更新本地这条群数据)
注意:写到这里,更准确的说,应该是先判断下自己还在不在members里面,如果在才更新,如果我自己不存在这个members里面了,说明群主把我移除了,不需要更新,直接delete这条数据库如果我现在正在这个聊天室,还应该退出当前聊天室,直接pop就可以了,因为pop到外面,数据库更新了,页面onShow一刷新显示最新的数据库,刚刚的群肯定不见了,因为本地数据库都在刚刚delete了。