如果你想做一个基于UDP包的数据处理问题的话,比较好的办法是使用csv文件来进行数据的保存(csv文件比较简单,方便进行各种处理)。
CSV文件格式简单,一方面它可以直接被excel处理,另一方面它完全由逗号和\0分隔,比较容易转化成Char[]这样的C语言中的变量。
主要功能为:
1.将打算发出UDP包的数据按顺序在csv文件中排好,并由C++开发的程序读取并发出。
2.将接收到的UDP数据按顺序写入一个csv文件之中。
也即,从一个CSV文件中读取你想要的行列数据赋值给某个变量。
1.如何选取csv文件?
2.如何读取csv文件?
3.如何处理csv字符使之可以UDP发送?
我们可以给定一个CSV看看效果,这里使用CFileDialog类选择文件,使用上述方法选择文件路径,并读取csv文件内容:
我们在如下rc中写代码:
下面说打开按钮
void CMFCCSVSENDDlg::OnBnClickedOpen()
{// TODO: 在此添加控件通知处理程序代码CFileDialog openDlg(TRUE,_T("CLS File(*.csv)|*.csv"),NULL,OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,_T("CLS File(*.csv)|*.csv||"),this);INT_PTR result = openDlg.DoModal();//以模态方式创建打开文件对话框if (result == IDOK)//如果有选中文件,那么result就是IDOK{OPEN_pathName = openDlg.GetPathName();OPEN_fileName = openDlg.GetFileName();MessageBox(OPEN_pathName + "\r" + OPEN_fileName + "\r" + "设置读取路径成功");}
}
上面这一段代码让你获得csv文件的路径,方便你打开它。
virtual BOOL Open(LPCTSTR lpszFileName,UINT nOpenFlags,CFileException* pError = NULL);virtual BOOL Open(LPCTSTR lpszFileName,UINT nOpenFlags,CAtlTransactionManager* pTM,CFileException* pError = NULL);
你可以使用MFCshellList功能配合button完成对路径lpszFileName的选择。
1.lpszFileName
一个字符串,它是所需文件的路径。 路径可以是相对路径或绝对路径。
2.nOpenFlags
共享和访问模式。 指定打开文件时要执行的操作。 可以使用按位“或”(|) 运算符来组合选项。 一个访问权限和一个共享选项是必需的;modeCreate 和 modeNoInherit 模式是可选的。
3.pError
指向接收失败操作状态的现有文件异常对象的指针。
4.pTM
指向 CAtlTransactionManager 对象的指针。
具体的,针对某一个csv文件,我们可以用如下办法提取CString:
void CMFCCSVSENDDlg::OnBnClickedSend()
{// TODO: 在此添加控件通知处理程序代码CStdioFile CSFile;CSFile.Open(OPEN_pathName, CFile::modeRead);CString C_buf_1;CSFile.ReadString(C_buf_1);MessageBox(C_buf_1, _T("C_buf_1"));
}
获得CSV中一行的字符串,csv文件的格式就是以’\n’与’,'对数据进行划分的。。
效果显示:
成功读取出了csv中的一行。
第一步是重新编辑分隔符:
CString x = _T(",");CString y = _T(" ");str.Replace(x,y);
第二步将CString转位String:
//CString转Stringsize_t i;int iSize;iSize = WideCharToMultiByte(CP_ACP, 0, str, -1, NULL, 0, NULL, NULL); //iSize =wcslen(pwsUnicode)+1=6char* pMBBuffer = (char*)malloc(iSize);wcstombs_s(&i, pMBBuffer, (size_t)iSize,str, (size_t)iSize - 1);
第三步将string录入sstream中,并给出为char[]:
istringstream is(pMBBuffer);char buff;char msg[100] = {0};int i = 0;while (is>>buff) {msg[i] = buff;i++;}CString box(msg);MessageBox(box);
注意使用istringstream时要在iostream、sstream和std namespace都在的时候调用:
#include
#include
using namespace std;
实现效果:
成功整合成一个我们想要的字符串。
为了更加清楚的展示,这里我们使用更进一步的方法,将CSV文件做成这样的:
重新执行上述代码,得到:
看上去,10和12、14都像是字符串,但其实他们已经是char本体了。
因为WindowsAPI往往并不真的需要你给定HEX格式的编码,更多的时候,WindowsAPI只需要你给定一个char[]或者一个string,如果你给定的是ASCII的string,虽然最后也是以HEX的格式发出,但意思完全不同。
以sendto函数为例,它其实本来就没有管你编码的事情:
sendto(sockfd, (char*)&msg, sizeof(msg), 0, (struct sockaddr*)&ServerAddr, sizeof(ServerAddr))
你直接给char[],它就已经可以发送了。
或者说,char[]本身就是一种string,string写出来的东西也就是一种char[]。
设定发送的IP地址和端口:
1.在类中建立变量Guest_CSocket;
2.在初始化中对Guest_CSocket初始化;
3.读取IP地址和PORT发送出去;
4.使用别的方式读取UDP包;
//打开回调
void CMFCCSVSENDDlg::OnBnClickedOpen()
{// TODO: 在此添加控件通知处理程序代码CFileDialog openDlg(TRUE,_T("CLS File(*.csv)|*.csv"),NULL,OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,_T("CLS File(*.csv)|*.csv||"),this);INT_PTR result = openDlg.DoModal();//以模态方式创建打开文件对话框if (result == IDOK)//如果有选中文件,那么result就是IDOK{OPEN_pathName = openDlg.GetPathName();OPEN_fileName = openDlg.GetFileName();MessageBox(OPEN_pathName + "\r" + OPEN_fileName + "\r" + "设置读取路径成功");}
}//保存回调
void CMFCCSVSENDDlg::OnBnClickedSave()
{// TODO: 在此添加控件通知处理程序代码 // TODO: 在此添加控件通知处理程序代码CFileDialog openDlg(FALSE,_T("CLS File(*.csv)|*.csv"),NULL,OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,_T("CLS File(*.csv)|*.csv||"),this);INT_PTR result = openDlg.DoModal();//以模态方式创建打开文件对话框if (result == IDOK)//如果有选中文件,那么result就是IDOK{SAVE_pathName = openDlg.GetPathName();SAVE_fileName = openDlg.GetFileName();MessageBox(SAVE_pathName + "\r" + SAVE_fileName + "\r" + "设置存储路径成功");}
}//发送回调
void CMFCCSVSENDDlg::OnBnClickedSend()
{// TODO: 在此添加控件通知处理程序代码CStdioFile CSFile;CSFile.Open(OPEN_pathName, CFile::modeRead);CString C_buf;CSFile.ReadString(C_buf);CString x = _T(",");CString y = _T(" ");C_buf.Replace(x, y);//CString to Stringsize_t i;int iSize;iSize = WideCharToMultiByte(CP_ACP, 0, C_buf, -1, NULL, 0, NULL, NULL); //iSize =wcslen(pwsUnicode)+1=6char* pMBBuffer = (char*)malloc(iSize);wcstombs_s(&i, pMBBuffer, (size_t)iSize,C_buf, (size_t)iSize - 1);//变为char[]istringstream is(pMBBuffer);char buff;char msg[100] = { 0 };int num = 0;while (is >> buff) {msg[num] = buff;num++;}//绑定IP与PORTCString C_G_IP;CString C_G_port;GUEST_IP_E.GetWindowText(C_G_IP);GUEST_PORT_E.GetWindowText(C_G_port);int G_port_i = _wtoi(C_G_port);CString C_H_IP;CString C_H_port;HOST_IP_E.GetWindowText(C_H_IP);HOST_PORT_E.GetWindowText(C_H_port);int H_port_i = _wtoi(C_H_port);if ((G_port_i < 10000)&&(H_port_i < 10000)){if (!Host_CSocket.Create(0, SOCK_DGRAM, NULL))//这里在正常使用时需要设定你输入的IP地址和PORT{MessageBox(_T("主机端套接字失败"));}else{if (Host_CSocket.SendTo((char*)&msg, sizeof(msg), G_port_i, C_G_IP, 0) != SOCKET_ERROR){CString str;str.Format(_T("数据发送成功"));MessageBox(str);}else{CString str;str.Format(_T("数据发送失败,套接字错误码 : %d"), GetLastError());MessageBox(str);}}}else{MessageBox(_T("端口设置错误"));}Host_CSocket.Close();
}
这里还有一个问题是如何发送HEX,这个问题下一篇博客讲。主要问题在于CString转数字的办法。
上一篇:C#基础教程21 正则表达式
下一篇:使用 KMS 驱动进行数据加密