IPMSG(飞秋)源代码调试笔记
1、怎么發送普通消息?
進行socket異步select選擇,每個消息都是一個Msg包,并且分配一個senddlg為其發送消息。消息協議可以參考其他文檔,消息統一封裝成文本的方式發送,封裝方式為[屬性1:屬性2:...:消息擴展消息],屬性和屬性之間和消息之間用冒號隔開,消息和擴展消息用消息結構體的exOffset標記隔開,發送的內容可以協定解密方式。
2、使用udp協議發送,如何確定消息是否已經發送成功?
消息發送出去,對方收到消息后會返回一條消息接收成功的確認消息,
然后發送端為每個senddlg開啟一個ontimer函數,每1500ms確定一次(時間可以自己配置)發送出去的消息是否收到的回應,如果未收到則重新發送。
3、接收端如何確定兩個消息是一樣的,而丟棄后面的?
判斷packetNo,判斷addr ,判斷 portNo,pocketNo是根據time()生成的。
4、msg 結構體(如下)的exOffset字段干啥用的?
struct MsgBuf {HostSub hostSub;int version;int portNo;ULONG packetNo;ULONG command;int exOffset; // expand message offset in msgBufint dummy;char msgBuf[MAX_UDPBUF];void Init(MsgBuf *org) {if (org == NULL) {memset(this, 0, (char *)&this->dummy - (char *)this);*msgBuf = 0;return;}memcpy(this, org, (char *)&this->dummy - (char *)this);strcpy(this->msgBuf, org->msgBuf);strcpy(this->msgBuf + exOffset, org->msgBuf + exOffset);} };??? exOffset表示可以發送除消息意外的其他內容(比如組名稱啥的,和消息區別開來),但前提是整個消息(加上exOffset)大小要小于MAX_UDPBUF。其實不用這個字段也行,我們可以把組名稱啥的封裝到消息buf里面去
5、如果輸入的字符串超出了MAX_UDPBUF怎么辦?
飛秋里面沒處理,直接截斷。。。
可以查看如下兩處處理方式:
(1)從編輯框獲取文本的時候,最大長度就用MAX_UDPBUF:
GetDlgItemText(SEND_EDIT, msg.msgBuf, MAX_UDPBUF);
(2)收取消息的時候,還會進行如下代碼判斷:
if (exStr && (ex_len = strlen(exStr) + 1) < MAX_UDPBUF -1)
{
if (++msg->exOffset + ex_len >= MAX_UDPBUF)
msg->msgBuf[(msg->exOffset = MAX_UDPBUF - ex_len) -1] = '\0';//這句表示超出直接截斷
memcpy(msg->msgBuf + msg->exOffset, exStr, ex_len);
}
6、在創建IPM的時候就進入了tcp_socket 就進入了listen的狀態。然后udp和tcp還可以同時使用一個端口,包括登入退出發送的廣播消息一并使用同一個端口,UDP發送消息,TCP發送文件,對socket的異步管理比較到位。
7、文件發送如何高效?
(1)使用內存映射文件的方式打開文件,內存映射文件方式能高效處理大文件。
(2)為發送和接收程序單獨開線程,并使用select選擇模型處理數據。
用內存映射方式打開關閉文件:
BOOL TMainWin::OpenSendFile(const char *fname, SendFileObj *obj) {DWORD lowSize, highSize, viewSize;if ((obj->hFile = ::CreateFile(fname, GENERIC_READ, FILE_SHARE_READ|FILE_SHARE_WRITE, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0)) != INVALID_HANDLE_VALUE){lowSize = ::GetFileSize(obj->hFile, &highSize);if ((obj->fileSize = (_int64)highSize << 32 | lowSize) == 0)return TRUE;obj->hMap = ::CreateFileMapping(obj->hFile, 0, PAGE_READONLY, highSize, lowSize, 0);viewSize = (int)(obj->fileSize > cfg->ViewMax ? cfg->ViewMax : obj->fileSize);highSize = (int)(obj->offset >> 32);lowSize = (int)((obj->offset / cfg->ViewMax) * cfg->ViewMax);obj->mapAddr = (char *)::MapViewOfFile(obj->hMap, FILE_MAP_READ, highSize, lowSize, viewSize);if (obj->mapAddr && IsBadReadPtr(obj->mapAddr, 1)){CloseSendFile(obj);return FALSE;}}return obj->mapAddr ? TRUE : FALSE; }BOOL TMainWin::CloseSendFile(SendFileObj *obj) {if (obj == NULL)return FALSE;::UnmapViewOfFile(obj->mapAddr);obj->mapAddr= NULL;::CloseHandle(obj->hMap); obj->hMap = NULL;::CloseHandle(obj->hFile); obj->hFile = INVALID_HANDLE_VALUE;obj->offset = 0;return TRUE; }?
發送文件線程:
DWORD WINAPI TMainWin::SendFileThread(void *_sendFileObj) {SendFileObj *obj = (SendFileObj *)_sendFileObj;fd_set fds;fd_set *rfds = NULL, *wfds = &fds;timeval tv;int sock_ret;BOOL ret = FALSE, completeWait = FALSE;BOOL (TMainWin::*SendFileFunc)(SendFileObj *obj) =obj->command == IPMSG_GETDIRFILES ? &TMainWin::SendDirFile : &TMainWin::SendFile;FD_ZERO(&fds);FD_SET(obj->conInfo->sd, &fds);for (int waitCnt=0; waitCnt < 180 && obj->hThread != NULL; waitCnt++){tv.tv_sec = 1, tv.tv_usec = 0;if ((sock_ret = ::select(obj->conInfo->sd + 1, rfds, wfds, NULL, &tv)) > 0){waitCnt = 0;if (completeWait){ // dummy read if (::recv(obj->conInfo->sd, (char *)&ret, sizeof(ret), 0) >= 0)ret = TRUE;break;}else if ((mainWin->*SendFileFunc)(obj) != TRUE)break;else if (obj->status == FS_COMPLETE){completeWait = TRUE, rfds = &fds, wfds = NULL;if (obj->fileSize == 0) { ret = TRUE; break; }}}else if (sock_ret == 0) {FD_ZERO(&fds);FD_SET(obj->conInfo->sd, &fds);}else if (sock_ret == SOCKET_ERROR) {break;}}if (obj->isDir){mainWin->CloseSendFile(obj);while (--obj->dirCnt >= 0)::FindClose(obj->hDir[obj->dirCnt]);}obj->status = ret ? FS_COMPLETE : FS_ERROR;mainWin->PostMessage(WM_TCPEVENT, obj->conInfo->sd, FD_CLOSE);::ExitThread(0);return 0; }BOOL TMainWin::SendFile(SendFileObj *obj) {if (obj == NULL || obj->hFile == INVALID_HANDLE_VALUE)return FALSE;int size = 0;_int64 remain = obj->fileSize - obj->offset;int transMax = cfg->TransMax - (int)(obj->offset % cfg->TransMax);if (remain > 0 && (size = ::send(obj->conInfo->sd, obj->mapAddr + (obj->offset % cfg->ViewMax), (int)(remain > transMax ? transMax : remain), 0)) < 0)return FALSE;obj->offset += size;if (obj->offset == obj->fileSize)obj->status = obj->command == IPMSG_GETDIRFILES ? FS_ENDFILE : FS_COMPLETE;else if ((obj->offset % cfg->ViewMax) == 0) // {::UnmapViewOfFile(obj->mapAddr);remain = obj->fileSize - obj->offset;obj->mapAddr = (char *)::MapViewOfFile(obj->hMap, FILE_MAP_READ, (int)(obj->offset >> 32), (int)obj->offset, (int)(remain > cfg->ViewMax ? cfg->ViewMax : remain));}obj->conInfo->lastTick = ::GetTickCount();return TRUE; }BOOL TMainWin::EndSendFile(SendFileObj *obj) {if (obj == NULL)return FALSE;if (obj->hThread){HANDLE hThread = obj->hThread;obj->hThread = 0; ::WaitForSingleObject(hThread, INFINITE);::CloseHandle(hThread);}if (::closesocket(obj->conInfo->sd) != 0)obj->status = FS_ERROR; CloseSendFile(obj);if (obj->isDir)free(obj->hDir);shareMng->EndHostShare(obj->packetNo, &obj->host->hostSub, obj->fileInfo, obj->status == FS_COMPLETE ? TRUE : FALSE);sendFileList.DelObj(obj);delete obj->conInfo;delete obj;return TRUE; }?
總結
以上是生活随笔為你收集整理的IPMSG(飞秋)源代码调试笔记的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: informix报错244_inform
- 下一篇: 工训计算机组维报告,计算机实训报告