c#中serialport发送数据给硬件的问题
我在局域网内用好几台机子通过UDP给我的程序发送数据,程序再将接收到的数据发给硬件,但serialport好像忙不过来自动退出,怎样修改serialport属性,使得能正...
我在局域网内用好几台机子通过UDP给我的程序发送数据,程序再将接收到的数据发给硬件,但serialport好像忙不过来自动退出,怎样修改serialport属性,使得能正常给硬件发数据?
展开
2013-09-02
展开全部
首先启动VC新建一个给予SDI的工程,然后加入SerialPort类。由于要进行多串口通讯,我们需要对SerialPort进行一些简单的修改,由于在与硬件通讯过程中一般通讯协议都采用BYTE类型数据传送,我们可以将改类中间的发送和接收数据类型修改成为BYTE类型。我修改了下面部分内容,详细改动请见附录提供的SERIALPORT类。
//
// Write a string to the port
//
void CSerialPort::WriteToPort(BYTE bWriteBuffer[],int nWriteBufferSize)
{
assert(m_hComm != 0);
int nSize = sizeof(bWriteBuffer)/sizeof(BYTE);
m_nWriteBufferSize = nWriteBufferSize;
for(int i = 0 ; i < nWriteBufferSize ; i ++)
m_bWriteBuffer[i] = bWriteBuffer[i]; // set event for write
SetEvent(m_hWriteEvent);
}
。。。。。
由于我们改串口接入了20台温度设备,在进行通讯的时候是通过发送某个地址的设备命令进行读取数据。我们首先对硬件设置相应的地址,这里我们设置0到19号地址。采集的时候采用循环的方式从0号地址向19号地址进行读取数据。当收到相应的数据包的时候我们进行相应的地址的数据解包处理。然后发送下一个地址的要数据命令。当地址为最后一台设备的时候我们将地址清0处理就可以了。但是如果我们这个20台设备中间某一个或者多个设备由于故障或者电源没开的话,上述通讯就会出现问题,我们发送没有运行的地址设备就会收不到相应的报文,我们就不会发送下一个地址的要数据命令,这是程序就会不走下去了。解决方法可以是我们从外部去判断是否对当前地址的发送要数据命令和收到数据命令是否超时。如果超时就进行跳过然后发送下一个地址要数据命令。当出现规定几个循环的时候进行该设备的采集参数清0等工作这个就可以随自己定义考虑了。具体实现如下:
定义SERIALPORT类对象,创建线程进行通讯。
CSerialPort m_Ports;
int nColtAddr,这个用来存放当前采集设备地址。nColts;这个用来存放当前缓冲区收到的字节数目
HANDLE m_pThread;外部控制线程
BYTE m_RecBuff[1000];接收缓冲区
float fVal[20];处理解包内容,这里可以根据实际情况进行定义。
启动串口监视线程和外部控制线程
nColtAddr = 0 ;
nColts = 0;
if(m_Ports.InitPort(this,1,4800,'N',8,1,EV_RXCHAR|EV_RXFLAG,1024))
{
this->m_Ports.StartMonitoring();启动监视线程
SetCommVal();发送第一台设备数据命令
}
下面是启动外部控制线程
unsigned int nDummy;
m_pThread=(HANDLE) _beginthreadex(NULL,0,CommThread,this,CREATE_SUSPENDED,&nDummy);//开辟外部控制线程
ResumeThread(m_pThread); 运行线程外部控制线程控制当前设备发送要数据命令和收到数据报文是否超时
UINT C××××View::CommThread(LPVOID pParam)
{
C××××View *pView = (C××××View *)pParam;
while(1)
{
CTime cNowTime = CTime::GetCurrentTime();
tNow = cNowTime.GetTime();
struct _timeb timebuffer;
_ftime(&timebuffer);
int nNowMillSecond = timebuffer.millitm;
///
tLast = cLastColtTime[0].GetTime();
if((tNow - tLast)*1000 + (nNowMillSecond - nMillSecond[0]) > 800)
pView->SetCommVal();发送下一台设备要数据命令或者进行其他的相关处理
Sleep(100);
}
}发送串口数据命令,这里要根据外部设备的制定的通讯协议来进行。这次温度表采用的是ASCII的形式通讯。
void C××××View::SetCommVal()
{
int HAddr,LAddr,m_Xnh;
int nHAdd,nLAdd;
nHAdd = ExchangeAscII((nColtAddr>>4)&0x0f);
nLAdd = ExchangeAscII(nColtAddr&0x0f);
m_Xnh = nHAdd^nLAdd^0x52^0x44;
HAddr = ExchangeAscII((m_Xnh>>4)&0x0f);
LAddr = ExchangeAscII(m_Xnh&0x0f);
BYTE OutBuff[8] = {0x40,nHAdd,nLAdd,0x52,0x44,HAddr,LAddr,0x0d};
m_Ports.WriteToPort(OutBuff,8);
cLastColtTime = CTime::GetCurrentTime();
nColtAddr++;
if(nColtAddr > 19)//19 define max addr numbers
nColtAddr = 0;
}ASCII码的一些简单变换,我们进行一下简单的封装,方便调用:
BYTE C××××View::ExchangeAscII(BYTE bInput)
{
BYTE bRef = 0;
if(bInput > 9)
bRef = bInput+0x37;
else
bRef = bInput+0x30;
return bRef;
}
BYTE C××××View::ExchangeAscIItoNormal(BYTE bInput)
{
BYTE bRef = 0;
if(bInput > 0x39)
bRef = bInput-0x37;
else
bRef = bInput-0x30;
return bRef;
}LONG C×××View::OnCommunication(WPARAM ch, LPARAM port)进行数据处理,WPARAM,LPARAM类型是多态性数据(polymorphic data type),在WIN32中为32位,支持多种数据类型,根据需要自动适应,这样程序就有很强的适应性。再次我们这里理解成为BYTE类型(与外围设备通讯协议保持一致,方便解包)。每当串口接收缓冲区内有一个字符的时候,就会产生一个WM_COMM_RXCHAR消息,触发OnCommunication函数,下面我们可以根据我们的需要进行解包处理了;
LONG CMy11View::OnCommunication(WPARAM ch, LPARAM port)
{
if(port == 1)
{
m_RecBuff[nColts] += (BYTE)(char *)(ch);
nColts++;
if(nColts == 24)这里根据通讯协议规定的发送定制要数据命令就会上传24个字节的数据报文内容。这里可以根据不同外部设备进行不同的设置
{
DataProcessTemp(m_RecBuff);处理解包
nColts = 0;缓冲区指针清0,准备接收下一台设备数据
ResetBuffVal();清空缓冲区内容
SetCommVal(); 发送下一台设备内容
}
}
return 0;
}
数据解包处理,这里就必须根据外部设备定义的通讯协议来处理了。
void CMy11View::DataProcessTemp(BYTE m_Inbuff[])
{
int nTempAddr = nColtAddr - 1;
if(nTempAddr < 0)
nTempAddr = 19;
int nHAdd,nLAdd;
nHAdd = ExchangeAscII((nTempAddr>>4)&0x0f);
nLAdd = ExchangeAscII(nTempAddr&0x0f);
if(m_Inbuff[0] == 0x40)
{
if(m_Inbuff[1] == nHAdd && m_Inbuff[2] == nLAdd)
{
if(m_Inbuff[3] == 0x52 && m_Inbuff[4] == 0x44)
{
int nzTemp[5];
float fTemp;
nzTemp[0] = m_Inbuff[7];
nzTemp[1] = m_Inbuff[8];
nzTemp[2] = m_Inbuff[9];
nzTemp[3] = m_Inbuff[10];
for(int i = 0 ; i < 4; i ++)
{
if(nzTemp[i] > 0x39)
nzTemp[i] -= 0x37;
else
nzTemp[i] -= 0x30;
}
fTemp=float(nzTemp[1]+(nzTemp[0]<<4)+(nzTemp[3]<<8)+(nzTemp[2]<<12))/10;
fVal[nTempAddr] = fTemp;
RedrawWindow();
}
}
}
}void CMy11View::ResetBuffVal()
{
for(int i=0;i<1000;i++)
m_RecBuff[i] = 0;
//
// Write a string to the port
//
void CSerialPort::WriteToPort(BYTE bWriteBuffer[],int nWriteBufferSize)
{
assert(m_hComm != 0);
int nSize = sizeof(bWriteBuffer)/sizeof(BYTE);
m_nWriteBufferSize = nWriteBufferSize;
for(int i = 0 ; i < nWriteBufferSize ; i ++)
m_bWriteBuffer[i] = bWriteBuffer[i]; // set event for write
SetEvent(m_hWriteEvent);
}
。。。。。
由于我们改串口接入了20台温度设备,在进行通讯的时候是通过发送某个地址的设备命令进行读取数据。我们首先对硬件设置相应的地址,这里我们设置0到19号地址。采集的时候采用循环的方式从0号地址向19号地址进行读取数据。当收到相应的数据包的时候我们进行相应的地址的数据解包处理。然后发送下一个地址的要数据命令。当地址为最后一台设备的时候我们将地址清0处理就可以了。但是如果我们这个20台设备中间某一个或者多个设备由于故障或者电源没开的话,上述通讯就会出现问题,我们发送没有运行的地址设备就会收不到相应的报文,我们就不会发送下一个地址的要数据命令,这是程序就会不走下去了。解决方法可以是我们从外部去判断是否对当前地址的发送要数据命令和收到数据命令是否超时。如果超时就进行跳过然后发送下一个地址要数据命令。当出现规定几个循环的时候进行该设备的采集参数清0等工作这个就可以随自己定义考虑了。具体实现如下:
定义SERIALPORT类对象,创建线程进行通讯。
CSerialPort m_Ports;
int nColtAddr,这个用来存放当前采集设备地址。nColts;这个用来存放当前缓冲区收到的字节数目
HANDLE m_pThread;外部控制线程
BYTE m_RecBuff[1000];接收缓冲区
float fVal[20];处理解包内容,这里可以根据实际情况进行定义。
启动串口监视线程和外部控制线程
nColtAddr = 0 ;
nColts = 0;
if(m_Ports.InitPort(this,1,4800,'N',8,1,EV_RXCHAR|EV_RXFLAG,1024))
{
this->m_Ports.StartMonitoring();启动监视线程
SetCommVal();发送第一台设备数据命令
}
下面是启动外部控制线程
unsigned int nDummy;
m_pThread=(HANDLE) _beginthreadex(NULL,0,CommThread,this,CREATE_SUSPENDED,&nDummy);//开辟外部控制线程
ResumeThread(m_pThread); 运行线程外部控制线程控制当前设备发送要数据命令和收到数据报文是否超时
UINT C××××View::CommThread(LPVOID pParam)
{
C××××View *pView = (C××××View *)pParam;
while(1)
{
CTime cNowTime = CTime::GetCurrentTime();
tNow = cNowTime.GetTime();
struct _timeb timebuffer;
_ftime(&timebuffer);
int nNowMillSecond = timebuffer.millitm;
///
tLast = cLastColtTime[0].GetTime();
if((tNow - tLast)*1000 + (nNowMillSecond - nMillSecond[0]) > 800)
pView->SetCommVal();发送下一台设备要数据命令或者进行其他的相关处理
Sleep(100);
}
}发送串口数据命令,这里要根据外部设备的制定的通讯协议来进行。这次温度表采用的是ASCII的形式通讯。
void C××××View::SetCommVal()
{
int HAddr,LAddr,m_Xnh;
int nHAdd,nLAdd;
nHAdd = ExchangeAscII((nColtAddr>>4)&0x0f);
nLAdd = ExchangeAscII(nColtAddr&0x0f);
m_Xnh = nHAdd^nLAdd^0x52^0x44;
HAddr = ExchangeAscII((m_Xnh>>4)&0x0f);
LAddr = ExchangeAscII(m_Xnh&0x0f);
BYTE OutBuff[8] = {0x40,nHAdd,nLAdd,0x52,0x44,HAddr,LAddr,0x0d};
m_Ports.WriteToPort(OutBuff,8);
cLastColtTime = CTime::GetCurrentTime();
nColtAddr++;
if(nColtAddr > 19)//19 define max addr numbers
nColtAddr = 0;
}ASCII码的一些简单变换,我们进行一下简单的封装,方便调用:
BYTE C××××View::ExchangeAscII(BYTE bInput)
{
BYTE bRef = 0;
if(bInput > 9)
bRef = bInput+0x37;
else
bRef = bInput+0x30;
return bRef;
}
BYTE C××××View::ExchangeAscIItoNormal(BYTE bInput)
{
BYTE bRef = 0;
if(bInput > 0x39)
bRef = bInput-0x37;
else
bRef = bInput-0x30;
return bRef;
}LONG C×××View::OnCommunication(WPARAM ch, LPARAM port)进行数据处理,WPARAM,LPARAM类型是多态性数据(polymorphic data type),在WIN32中为32位,支持多种数据类型,根据需要自动适应,这样程序就有很强的适应性。再次我们这里理解成为BYTE类型(与外围设备通讯协议保持一致,方便解包)。每当串口接收缓冲区内有一个字符的时候,就会产生一个WM_COMM_RXCHAR消息,触发OnCommunication函数,下面我们可以根据我们的需要进行解包处理了;
LONG CMy11View::OnCommunication(WPARAM ch, LPARAM port)
{
if(port == 1)
{
m_RecBuff[nColts] += (BYTE)(char *)(ch);
nColts++;
if(nColts == 24)这里根据通讯协议规定的发送定制要数据命令就会上传24个字节的数据报文内容。这里可以根据不同外部设备进行不同的设置
{
DataProcessTemp(m_RecBuff);处理解包
nColts = 0;缓冲区指针清0,准备接收下一台设备数据
ResetBuffVal();清空缓冲区内容
SetCommVal(); 发送下一台设备内容
}
}
return 0;
}
数据解包处理,这里就必须根据外部设备定义的通讯协议来处理了。
void CMy11View::DataProcessTemp(BYTE m_Inbuff[])
{
int nTempAddr = nColtAddr - 1;
if(nTempAddr < 0)
nTempAddr = 19;
int nHAdd,nLAdd;
nHAdd = ExchangeAscII((nTempAddr>>4)&0x0f);
nLAdd = ExchangeAscII(nTempAddr&0x0f);
if(m_Inbuff[0] == 0x40)
{
if(m_Inbuff[1] == nHAdd && m_Inbuff[2] == nLAdd)
{
if(m_Inbuff[3] == 0x52 && m_Inbuff[4] == 0x44)
{
int nzTemp[5];
float fTemp;
nzTemp[0] = m_Inbuff[7];
nzTemp[1] = m_Inbuff[8];
nzTemp[2] = m_Inbuff[9];
nzTemp[3] = m_Inbuff[10];
for(int i = 0 ; i < 4; i ++)
{
if(nzTemp[i] > 0x39)
nzTemp[i] -= 0x37;
else
nzTemp[i] -= 0x30;
}
fTemp=float(nzTemp[1]+(nzTemp[0]<<4)+(nzTemp[3]<<8)+(nzTemp[2]<<12))/10;
fVal[nTempAddr] = fTemp;
RedrawWindow();
}
}
}
}void CMy11View::ResetBuffVal()
{
for(int i=0;i<1000;i++)
m_RecBuff[i] = 0;
推荐律师服务:
若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询