logo
当前页

房间管理


功能简介

ZIM SDK 提供多人房间聊天功能,支持用户向房间内发送文本消息或自定义消息,实现了多人在线交流、同步分享。

多人房间聊天功能可应用于小班课或者会议室等场景,房间成员数量上限请参考 计费说明

实现流程

用户可以通过以下两种方式,创建房间并进入房间。

房间内的用户,可以通过 sendRoomMessage 接口,向房间内发送消息,详情请参考 收发房间消息

说明

如果 roomID 已存在:

如果 roomID 不存在:

  • 调用 createRoom 接口,可以直接创建、并加入到此房间内。
  • 调用 enterRoom 接口,会直接创建一个房间、并进入到此房间内。

创建房间、加入房间

以下流程中,我们以客户端 A 创建并加入房间,客户端 B 加入房间为例。

1. 注册回调

所有进入房间的用户,都需注册 onRoomMemberJoinedonRoomMemberLeftonRoomStateChanged 回调,用于监听房间其他成员的变动和接收房间连接状态发生改变的事件通知。

注册 onRoomMemberJoined 回调示例源码:

Untitled
void onRoomMemberJoined(ZIM * /*zim*/, const std::vector<ZIMUserInfo> & /*memberList*/,
    const std::string & /*roomID*/) override{
        
    }
1
Copied!

注册 onRoomMemberLeft 回调示例源码:

Untitled
void onRoomMemberLeft(ZIM * /*zim*/, const std::vector<ZIMUserInfo> & /*memberList*/,
    const std::string & /*roomID*/) override{
        
    }
1
Copied!

注册 onRoomStateChanged 回调示例源码:

Untitled
void onRoomStateChanged(ZIM * /*zim*/, ZIMRoomState /*state*/, ZIMRoomEvent /*event*/,
    const std::string & /*extendedData*/,
    const std::string & /*roomID*/)override{
        
    }
1
Copied!

2. 创建房间

客户端 A 登录后,创建一个房间,可以调用 createRoom 接口,传入 ZIMRoomInfo 信息,即可创建并加入房间。同时可以通过错误码 ZIMError 的参数值,判断是否创建成功。相关错误码请查看 常见错误码

创建成功后会收到 onRoomStateChanged 的通知回调, ZIMRoomStateZIM_ROOM_STATE_CONNECTED, ZIMRoomEventZIM_ROOM_EVENT_SUCCESS

注意
  • “roomID”、“roomName” 支持开发者自定义规则生成。建议开发者将 “roomID” 设置为一个有意义的值,可将其与自己的业务账号系统进行关联。
  • 调用 createRoom 接口创建房间后,会直接加入房间,无需再调用 joinRoom 接口加入房间。
Untitled
// roomID 最大 32 字节的字符串。仅支持数字,英文字符 和 '!', '#', '$', '%', '&', '(', ')', '+', '-', ':', ';', '<', '=', '.', '>', '?', '@', '[', ']', '^', '_', '{', '}', '|', '~'。
// roomName 最大 64 字节的字符串,无特殊字符限制。
ZIMRoomInfo room_info;
room_info.roomID = room_id;
room_info.roomName = room_name;

zim_->createRoom(room_info, [=](ZIMRoomFullInfo room_info, zim::ZIMError error_info) {
    global_main_dialog_->PostUiThread([=] {});
    if (error_info.code != 0)
    {
        ShowMsg(L"创建房间失败,房间ID: %s, 错误码: %d", room_info.baseInfo.roomID, error_info.code);
    }
    else
    {
        ShowMsg(L"创建房间成功,房间ID: %s", room_info.baseInfo.roomID);
    }
});
1
Copied!

3. 加入房间

客户端 B 加入房间,可以调用 joinRoom 接口,传入由 A 创建的房间 roomID,即可加入房间。同时可以根据错误码 ZIMError 的参数值,判断用户是否加入成功。相关错误码请查看 常见错误码

加入成功后会收到 onRoomStateChanged 的通知回调, ZIMRoomStateZIM_ROOM_STATE_CONNECTED , ZIMRoomEventZIM_ROOM_EVENT_SUCCESS

Untitled
zim_->joinRoom(roomId, [=](ZIMRoomFullInfo room_info, zim::ZIMError error_info) {
    global_main_dialog_->PostUiThread([=] {});
    if (error_info.code != 0)
    {
        ShowMsg(L"加入房间失败,房间ID:%s,错误代码:%d", room_info.baseInfo.roomID, error_info.code);
    }
    else
    {
        ShowMsg(L"成功加入房间,房间ID:%s,房间名称:%s", room_info.baseInfo.roomID, room_info.baseInfo.roomName);
    }
});
1
Copied!

4. 房间成员变动通知

当房间有其他成员加入时,可以使用 setEventHandler 回调接口 onRoomMemberJoined 方法,实现对其他房间成员加入的监听。

例如,当客户端 B 加入由 A 创建的房间时,A 将收到房间内成员变动的通知。

Untitled
void onRoomMemberJoined(ZIM* zim, const std::vector<ZIMUserInfo>& member_list, const std::string& room_id)
{
    for (auto member : member_list)
    {
        ShowMsg(L"用户: (%s%s) 进入房间: %s", member.userID, member.userName, room_id);
    }
}
1
Copied!

进入房间

以下流程中,我们以客户端 X 创建并进入房间,客户端 Y 直接进入房间为例。

  1. 注册 onRoomMemberJoinedonRoomMemberLeftonRoomStateChanged 回调,详情请参考 创建房间、加入房间 - 1. 注册回调
  2. 客户端 X 登录后,调用 enterRoom 接口,传入 ZIMRoomInfo 信息,进入房间;如果传入的 roomID 不存在,将会自动创建一个房间并进入该房间。
  3. 客户端 Y 登录后,调用 enterRoom 接口,传入由 X 创建的房间 roomID,直接进入房间。
  4. 进入房间成功后会收到 onRoomStateChanged 的通知回调, ZIMRoomStateZIM_ROOM_STATE_CONNECTED , ZIMRoomEventZIM_ROOM_EVENT_SUCCESS
  5. 房间内的用户,同样可以使用 setEventHandler 的回调接口 onRoomMemberJoined 方法,实现对房间内成员加入的监听。
Untitled
ZIMRoomInfo room_info;
room_info.roomID = room_id;
room_info.roomName = room_name;

ZIMRoomAdvancedConfig config;

zim_->enterRoom(room_info, config, [=](ZIMRoomFullInfo room_info, zim::ZIMError error_info) {
    global_main_dialog_->PostUiThread([=] {});
    if (error_info.code != 0)
    {
        ShowMsg(L"加入房间失败。房间ID:%s,错误代码:%d", room_info.baseInfo.roomID, error_info.code);
    }
    else
    {
        ShowMsg(L"成功加入房间。房间ID:%s", room_info.baseInfo.roomID);
    }
});
1
Copied!

切换房间

如果用户想要从一个房间切换至另一个房间,可以调用 switchRoom 接口,传入原房间的 ID(fromRoomID)以及目标房间信息(toRoomInfo,含房间 ID 和房间名称),即可切换房间。

然而,可能因目标房间不存在而导致房间切换失败。如需避免这种失败,也可以在调用 switchRoom 接口时,传入 isCreateWhenRoomNotExistedtrue(允许 ZIM 在房间不存在时创建新房间)和 config (用于配置新房间)。如此,当 ZIM 判断目标房间不存在时,就会根据 toRoomInfoconfig(如有)创建新房间用于切换。

房间切换成功后,原房间内的其他用户可通过 setEventHandler 的回调接口 onRoomMemberLeft ,得知有用户离开房间;目标房间内的其他用户可通过 setEventHandler 的回调接口 onRoomMemberJoined ,得知有用户进入了房间。

  • 完整流程请参考下图:

    %%{ init: { 'flowchart': { 'curve': 'stepAfter' } } }%% flowchart TD Start([开始]) --> A[用户 A 通过 switchRoom 从 Room1 切换至 Room2] A --> B[用户 A 通过 onRoomStateChanged 事件得知正在连接 Room2] B --> C{Room2 是否存在} C -->|否| D{服务器内房间数量\n是否达到上限?} C -->|是| E[房间切换成功\nRoom2 用户收到 onRoomMemberJoined 回调] D -->|是| K[房间切换失败\n用户 A 通过 onRoomStateChanged 事件得知与 Room2 断连] D -->|否| F{isCreateWhen\nRoomNotExisted \n参数是否为 true} F -->|否| K F -->|是| G[创建新房间 Room2,房间切换成功\nRoom2 各参数与调用 switchRoom 传入时一致] G --> H[用户 A 收到 2 次onRoomStateChanged 回调\n分别得知已连接 Room2 并与 Room1 断连] E --> H H --> I{Room1 是否\n有其他成员} I -->|是| J[Room1 用户收到 onRoomMemberLeft 回调] I -->|否| L[Room1 在延迟销毁时间到达时被销毁] J --> End([结束]) L --> End K --> End

    如上图所示,用户在切换房间的时候会接收到数次 onRoomStateChanged 事件通知,根据触发时机,该事件中相关参数如下表所示:

    触发时机
    roomID 值
    state 值
    event 值
    事件意义
    调用 switchRoom 接口后。Room2ZIM_ROOM_STATE_CONNECTINGZIM_ROOM_EVENT_ACTIVE_SWITCH正在连接 Room2。
    调用 switchRoom 接口后,Room2 不存在时,以下任意情况:
    • 服务器内房间数量已达上限。
    • 服务器内房间数量未达上限,但调用 switchRoom 接口时 isCreateWhenRoomNoteExisted 参数为 false。
    Room2ZIM_ROOM_STATE_DISCONNECTEDZIM_ROOM_EVENT_SWITCH_FAILED因 Room2 不存在导致切换失败。
    在以下任意情况下,用户会收到两次回调:
    • 调用 switchRoom 接口后,Room2 存在。
    • 调用 switchRoom 接口后,Room2 不存在,服务器内房间数量未达上限,且调用 switchRoom 接口时 isCreateWhenRoomNoteExisted 参数为 true。
    • 第一次回调:Room2
    • 第二次回调:Room1
    • 第一次回调:ZIM_ROOM_STATE_CONNECTED
    • 第二次回调:ZIM_ROOM_STATE_DISCONNECTED
    • 第一次回调:ZIM_ROOM_EVENT_SUCCESS
    • 第二次回调:ZIM_ROOM_EVENT_SUCCESS
    • 第一次回调:成功连接 Room2。
    • 第二次回调:成功断连 Room1。
  • 示例代码:

Untitled
std::string fromRoomID = "fromRoomID";
auto toRoomInfo = zim::ZIMRoomInfo("toRoomID", "toRoomName");

// 当房间不存在时是否创建房间。
// 只有当此值为true且目标房间不存在时,toRoomName和toRoomInfo中的config才有意义。
bool isCreateWhenRoomNotExisted = true; 

// 创建房间的高级配置
auto config = zim::ZIMRoomAdvancedConfig();
config.roomAttributes.emplace("key1", "value1");
config.roomDestroyDelayTime = 90;

// 切换房间
zim->switchRoom(
    fromRoomID, toRoomInfo, isCreateWhenRoomNotExisted, config,
    [=](const zim::ZIMRoomFullInfo &roomInfo, const zim::ZIMError &errorInfo) {
        if (errorInfo.code == zim::ZIMErrorCode::ZIM_ERROR_CODE_SUCCESS) {
            // 房间切换成功,您可以从roomInfo中获取有关切换后房间的信息
        }
    });
1
Copied!

离开房间

用户如果想要离开房间,可以调用 leaveRoom 接口,传入房间的 roomID,即可退出此房间;房间内的其他用户可以通过 setEventHandler 的回调接口 onRoomMemberLeft ,收到成员变动通知。

离开房间成功后会收到 onRoomStateChanged 的通知回调, ZIMRoomStateZIM_ROOM_STATE_DISCONNECTEDZIMRoomEventZIM_ROOM_EVENT_SUCCESS

离开房间后,将不能收到房间内的消息。

Untitled
zim_->leaveRoom(room_id, [=](zim::ZIMError error_info) {
    global_main_dialog_->PostUiThread([=] {
        ShowMsg(L"退出房间结果:%d", error_info.code);
    });
});
1
Copied!
Untitled
void onRoomMemberLeft(ZIM* zim, const std::vector<ZIMUserInfo>& member_list, const std::string& room_id)
{
    for (auto member : member_list)
    {
        ShowMsg(L"用户: (%s,%s) 退出房间: %s", member.userID, member.userName, room_id);
    }
}
1
Copied!
说明

当所有成员离开房间后,房间将自动销毁。ZIM SDK 支持通过 config 设置房间延迟销毁,最大延迟为 3 小时。

退出所有房间

调用 leaveAllRoom 接口,即可立即离开当前加入的所有房间,并返回离开的房间列表。如果用户多端登录,则用户在一端调用此接口后,仅会退出在该端上加入的房间,在其他端加入的房间不受影响。

Untitled
zim->leaveAllRoom([](const std::vector<std::string>& roomIDList, const ZIMError& errorInfo) {
    if (errorInfo.errorCode == 0) {
        // `roomIDList` 包含用户已离开的所有房间的ID。
    }else{
        // 根据相应的错误代码表处理错误。
    }
});
1
Copied!

房间内的其他用户可以通过 setEventHandler 的回调接口 onRoomMemberLeft ,收到成员变动通知。

退出所有房间成功后,用户会根据所离开房间的数量,多次收到 onRoomStateChanged 的通知回调, ZIMRoomStateZIM_ROOM_STATE_DISCONNECTEDZIMRoomEventZIM_ROOM_EVENT_SUCCESS

离开房间后,将不能收到房间内的消息。

销毁房间

开发者可以在后台调用 ZIM 服务端接口进行 销毁房间。房间被销毁后,用户通过 onRoomStateChanged 收到房间状态( ZIMRoomState )从 ZIM_ROOM_STATE_CONNECTED 转变为 ZIM_ROOM_STATE_DISCONNECTED 的事件通知。根据 ZIM SDK 版本不同,回调中描述导致房间连接状态发生变更的事件( ZIMRoomEvent )如下所示:

  • 当 ZIM SDK 版本小于 2.13.0 时,事件为 ZIM_ROOM_EVENT_KICKED_OUT
  • 当 ZIM SDK 版本为 2.13.0 及之后版本时,事件为 ZIM_ROOM_EVENT_ROOM_NOT_EXIST

网络中断对房间生命周期的影响

ZIM 房间类似于在线聊天室、网络中断会导致房间进入重连状态并抛出 onRoomStateChanged 。详情请参考 场景 4 : ZIM 房间断网

Previous

在线状态订阅

Next

房间资料管理