面临的困难
如果你想做一个基于UDP包的数据处理问题的话,比较好的办法是使用csv文件来进行数据的保存(csv文件比较简单,方便进行各种处理)。
CSV文件格式简单,一方面它可以直接被excel处理,另一方面它完全由逗号和\0分隔,比较容易转化成Char[]这样的C语言中的变量。
主要功能为:
1.将打算发出UDP包的数据按顺序在csv文件中排好,并由C++开发的程序读取并发出。
2.将接收到的UDP数据按顺序写入一个csv文件之中。
发送实现方法
也即,从一个CSV文件中读取你想要的行列数据赋值给某个变量。
1.如何选取csv文件?
2.如何读取csv文件?
3.如何处理csv字符使之可以UDP发送?
CFileDialog类(选择CSV文件)
我们可以给定一个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文件的路径,方便你打开它。
CStdioFile类(读取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中的一行。
处理csv文件变为char[]
第一步是重新编辑分隔符:
CString x = _T(",");
CString y = _T(" ");
str.Replace(x,y);
第二步将CString转位String:
//CString转String
size_t i;
int iSize;
iSize = WideCharToMultiByte(CP_ACP, 0, str, -1, NULL, 0, NULL, NULL); //iSize =wcslen(pwsUnicode)+1=6
char* 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 <iostream>
#include <sstream>
using namespace std;
实现效果:
成功整合成一个我们想要的字符串。
为了更加清楚的展示,这里我们使用更进一步的方法,将CSV文件做成这样的:
重新执行上述代码,得到:
看上去,10和12、14都像是字符串,但其实他们已经是char本体了。
为什么要处理成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 String
size_t i;
int iSize;
iSize = WideCharToMultiByte(CP_ACP, 0, C_buf, -1, NULL, 0, NULL, NULL); //iSize =wcslen(pwsUnicode)+1=6
char* 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与PORT
CString 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转数字的办法。