这次来个有界面的。

  登录界面:

  主界面:

1、服务端开发:

  只需创建一个类ChatType(./examples/HelloKhala/src/ChatType.cpp),在该类型中,核心为创建的两个消息事件,一个为onCurrFriends,用于获取当前在线用户;一个为onSendMsg,用于向具体用户发送消息。

  客户端请求当前的在线用户的信息,发送请求消息。该请求消息由onCurrFriends解析。onCurrFriends首先获取所有类型为ChatType的在线用户的id,然后再遍历这些id,根据id获取具体的用户信息,并将这些信息以json形式进行处理,设置类型为CHAT_TYPE,并最终将该json消息发送给请求的客户端。

bool ChatType::onCurrFriends(khala::InfoNodePtr& infoNodePtr, Json::Value& msg,
khala::Timestamp time) {
Json::Value res;
//返回给请求用户的消息类型
res[MSG_TYPE] = CHAT_FRIENDS;
res["result"] = "true";
khala::NodeManager* nodeManager = getNodeManager();
//获取类型为ChatType的在线用户的id
std::vector<uint> currIds = nodeManager->getNodeIDs(
infoNodePtr->getNodeType());
for (std::vector<uint>::iterator it = currIds.begin(); it != currIds.end();
++it) {
khala::InfoNodePtr usrInfoNode;
//根据id获取具体用户的信息
if (nodeManager->find(*it, usrInfoNode)) {
Json::Value usrJs;
usrJs[KEY_ID] = *it;
UsrInfo* usrInfo = boost::any_cast<UsrInfo*>(
usrInfoNode->getExtraContext());
usrJs["name"] = usrInfo->getName();
res["data"].append(usrJs);
}
}
//发送json消息
Json::FastWriter jwriter;
std::string sendStr = jwriter.write(res);
infoNodePtr->send(sendStr);
return true;
}

  客户端用户A请求发送消息msg给用户B,该请求消息由onSendMsg解析。onSendMsg获取消息msg,根据用户B的id获取用户B的对象,并将msg以json进行消息处理后设置消息类型为CHAT_REV,并发送给用户B。同时记录发送结果,并将结果以json形式进行处理,最后设置消息类型为CHAT_SEND将该结果返回给用户A。

bool ChatType::onSendMsg(khala::InfoNodePtr& infoNodePtr, Json::Value& msg,
khala::Timestamp time) {
Json::Value res;
//返回给请求发送用户的消息类型
res[MSG_TYPE] = CHAT_SEND;
//获取目的用户的id
uint friendId = msg[KEY_FRIEND_ID].asUInt();
//得到待发送消息
std::string sendMsg = msg[CHAT_MSG].asString();
res[FRIEND_NAME] = msg[FRIEND_NAME].asString();
res[CHAT_MSG] = sendMsg;
khala::InfoNodePtr friendNodePtr;
//根据目的用户的id获取目的用户的对象
if (this->getNodeManager()->find(friendId, friendNodePtr)) {
Json::Value sendJs;
//发送给目的用户的消息类型
sendJs[MSG_TYPE] = CHAT_REV;
//发送给目的用户的消息
sendJs[CHAT_MSG] = sendMsg;
UsrInfo* usrInfo = boost::any_cast<UsrInfo*>(
infoNodePtr->getExtraContext());
if (usrInfo == ) {
return false;
}
//发送用户的姓名和id
sendJs[SEND_NAME] = usrInfo->getName();
sendJs[KEY_FRIEND_ID] = infoNodePtr->getId();
//将该json消息发送给目的用户
Json::FastWriter jwriter;
std::string sendStr = jwriter.write(sendJs);
friendNodePtr->send(sendStr);
//向请求用户标示结果为成功
res["result"] = "true";
} else {
//向请求用户标示结果为失败
res["result"] = "false";
}
//将发送结果返回给请求用户
Json::FastWriter jwriter;
std::string sendStr = jwriter.write(res);
infoNodePtr->send(sendStr);
return true;
}

2、客户端开发:

  主要工作,拖界面,写控件响应事件!(废话)

  主要创建2个线程,一个为消息发送线程,一个为消息接收线程。

  消息发送线程:建立一个多生产者单消费者模型,每个窗口作为一个生产者,消息发送线程作为单独的消费者,选择System.Collections.Queue作为中间缓存。窗口将待发送json消息通过生产者接口发送给缓存容器。一旦缓存容器中存在待发送的消息,发送线程便被唤醒,从缓存中读取消息,对消息进行处理(选择utf8对字符进行处理,并添加长度头解决粘包问题),最终通过套接字发送给服务端。

  消息接收线程:以阻塞方式通过套接字循环接收消息,并将消息以字节形式保存在接收缓存中,通过长度头从接收缓存中获取每条完整的消息。再对该消息进行字符处理,解析为json形式,并根据消息类型将完整的消息通过SendMessage()发送给相应的窗口句柄进行处理。

  在每个具体窗口中,通过重写DefWndProc()来获取接收线程获取到的具体消息,并根据接收到的json消息编写窗口响应函数。

  具体客户端的代码就不贴了,不会C#,代码写得丑…

  最近应该会重写设备生命周期部分的代码,之前生命周期的超时检测部分设计实现得太丑陋,自己都看不下去了。争取少改动主要接口吧,阿门...

最新文章

  1. C#多线程之线程同步篇1
  2. checkbox判断选中
  3. poj 3228(二分+最大流)
  4. RequestHander类介绍
  5. (转) Unity3D中角色的动画脚本的编写(三)
  6. JS子元素oumouseover触发父元素onmouseout
  7. 关于Meta标签中format-detection属性及含义
  8. mac 上安装 nvm 遇到的坑
  9. thinkphp5+vue+iview商城 公众号+小程序更新版本
  10. docker 进阶
  11. Python_程序实现发红包
  12. mycat - 水平分表
  13. 禁止 pdf 下载、打印的方法
  14. JS window与document
  15. ubuntu 16.04 安装pgadmin3
  16. [转帖]Windows 使用netsh 命令行方式处理 windows防火墙的方法
  17. 转 linux下php扩展mysqli的支持
  18. Laravel 开启跨域请求
  19. window消息机制二
  20. HBase查询优化——持续更新

热门文章

  1. Struts2 技术全总结 (正在更新)
  2. OpenGL ES 2.0 剪裁测试
  3. Trident内核中取验证码图片的几种方法
  4. jquery 获取多个dom对象的方法
  5. python 学习(三)
  6. 异步编程设计模式Demo - AsyncComponentSample
  7. Android-3 Activity启动模式
  8. mysql设置连接等待时间(wait_timeout)
  9. [iOS常见问题] 关于使用QQ做第三方登录的问题!
  10. Ceph的Block分析