串口是计算机上一种非常通用设备通信的协议。大多数计算机包含两个基于RS232的串口。串口同时也是仪器仪表设备通用的通信协议;很多GPIB兼容的设备也带有RS-232口。同时,也可以用于获取远程采集设备的数据。
串口通信的概念非常简单,串口按位(bit)发送和接收字节。尽管比按字节(byte)的并行通信慢,但是串口可以在使用一根线发送数据的同时用另一根线接收数据。它很简单并且能够实现远距离通信。比如IEEE488定义并行通行状态时,规定设备线总长不得超过20米,并且任意两个设备间的长度不得超过2米;而对于串口而言,长度可达1200米。典型地,串口用于ASCII码字符的传输。通信使用3根线完成:(1)地线,(2)发送,(3)接收。由于串口通信是异步的,端口能够在一根线上发送数据同时在另一根线上接收数据。其他线用于握手,但是不是必须的。串口通信最重要的参数是波特率、数据位、停止位和奇偶校验。对于两个进行通行的端口,这些参数必须匹配:
a,波特率:这是一个衡量通信速度的参数。它表示每秒钟传送的bit的个数。例如300波特表示每秒钟发送300个bit。当我们提到时钟周期时,我们就是指波特率例如如果协议需要4800波特率,那么时钟是4800Hz。这意味着串口通信在数据线上的采样率为4800Hz。通常电话线的波特率为14400,28800和36600。波特率可以远远大于这些值,但是波特率和距离成反比。高波特率常常用于放置的很近的仪器间的通信,典型的例子就是GPIB设备的通信。
b,数据位:这是衡量通信中实际数据位的参数。当计算机发送一个信息包,实际的数据不会是8位的,标准的值是5、7和8位。如何设置取决于你想传送的信息。比如,标准的ASCII码是0~127(7位)。扩展的ASCII码是0~255(8位)。如果数据使用简单的文本(标准 ASCII码),那么每个数据包使用7位数据。每个包是指一个字节,包括开始/停止位,数据位和奇偶校验位。由于实际数据位取决于通信协议的选取,术语“包”指任何通信的情况。
c,停止位:用于表示单个包的{zh1}一位。典型的值为1,1.5和2位。由于数据是在传输线上定时的,并且每一个设备有其自己的时钟,很可能在通信中两台设备间出现了小小的不同步。因此停止位不仅仅是表示传输的结束,并且提供计算机校正时钟同步的机会。适用于停止位的位数越多,不同时钟同步的容忍程度越大,但是数据传输率同时也越慢。
d,奇偶校验位:在串口通信中一种简单的检错方式。有四种检错方式:偶、奇、高和低。当然没有校验位也是可以的。对于偶和奇校验的情况,串口会设置校验位(数据位后面的一位),用一个值确保传输的数据有偶个或者奇个逻辑高位。例如,如果数据是011,那么对于偶校验,校验位为0,保证逻辑高的位数是偶数个。如果是奇校验,校验位位1,这样就有3个逻辑高位。高位和低位不真正的检查数据,简单置位逻辑高或者逻辑低校验。这样使得接收设备能够知道一个位的状态,有机会判断是否有噪声干扰了通信或者是否传输和接收数据是否不同步。
RS-232(ANSI/EIA-232标准)是IBM-PC及其兼容机上的串行连接标准。可用于许多用途,比如连接鼠标、打印机或者Modem,同时也可以接工业仪器仪表。用于驱动和连线的改进,实际应用中RS-232的传输长度或者速度常常超过标准的值。RS-232只限于PC串口和设备间点对点的通信。RS-232串口通信最远距离是50英尺。
Cserial 是由MuMega Technologies公司提供的一个免费的VC++类,可方便地实现串行通信。最近搞串口程序,一般都是利用这个类,的确很方便
串行通信类Cserial 成员函数简介
1. CSerial::Cserial是类构造函数,不带参数,负责初始化所有类成员变量。
2. CSerial:: Open这个成员函数打开通信端口。带两个参数,{dy}个是埠号,有效值是1到4,第二个参数是波特率,返回一个布尔量。
3. CSerial:: Close函数关闭通信端口。类析构函数调用这个函数,所以可不用显式调用这个函数。
4. CSerial:: SendData函数把数据从一个缓冲区写到串行端口。它所带的{dy}个参数是缓冲区指针,其中包含要被发送的资料;这个函数返回已写到端口的实际字节数。
5. CSerial:: ReadDataWaiting函数返回等待在通信端口缓冲区中的数据,不带参数。
6. CSerial:: ReadData函数从端口接收缓冲区读入数据。{dy}个参数是void*缓冲区指针,资料将被放入该缓冲区;第二个参数是个整数值,给出缓冲区的大小。
ID卡读卡程序举例
void CSerialportDlg::OnButton1()
{
//serialport.Close();
/* if(!idcomport.IsOpened())
{
if(!idcomport.Open(1,9600,2))
{
AfxMessageBox("打开串口1失败!");
//return ;
}
AfxMessageBox("打开串口1成功!");
}
if(!upscomport.IsOpened())
{
if(!upscomport.Open(2,9600,2))
{
AfxMessageBox("打开串口2失败!");
return ;
}
AfxMessageBox("打开串口2成功!");
}
if(!idcomport.IsOpened())
{
if(!idcomport.Open(3,9600,2))
{
AfxMessageBox("打开串口3失败!");
return ;
}
else
AfxMessageBox("打开串口3成功!");
}
*/
if(!idcomport.IsOpened())
{
if(!idcomport.Open(4,9600,2))
{
AfxMessageBox("打开串口4失败!");
return ;
}
AfxMessageBox("打开串口4成功!");
}
/* if(!upscomport.IsOpened())
{
if(!upscomport.Open(4,9600,2))
{
AfxMessageBox("打开串口4失败!");
return ;
}
AfxMessageBox("打开串口4成功!");
}
*/
idcomport.ReadDataWaiting();
// upscomport.ReadDataWaiting();
MessageBox("请刷卡!");
hThread=CreateThread(NULL,0,(LPTHREAD_START_ROUTINE)Thread_IO,NULL,0,&ThreadID);
GetCRC(6,senbuf);
}
void Thread_IO()
{
while(1)
{
if((bbb=idcomport.ReadData(buffer,16))!=0)
{
m=9;
result=0;
// if((buffer[0]==63)&&(buffer[15]==0))
// {
for(int i=1;i<10;i+=2)
{
for(int j=0;j<2;j++)
{
a[j]=buffer[i+j];
if(a[j]>=48&&a[j]<=57)
a[j]=a[j]-48;
else
if(a[j]>=0x41&&a[j]<=0x46)
a[j]=data[a[j]-0x41];
}
result=result+a[0]*pow(16,m--)+a[1]*pow(16,m--);
}
str.Format("%d",result);
str="0"+str;
AfxMessageBox( "您测试的卡号是: " + str );
//利用卡号从数据库得到这个人的名字(利用RecordSet),然后插入表attendtmp中
/* _bstr_t sql="select *from staff where physicid='"+str+"'";
try{
recordset.CreateInstance("ADODB.Recordset");
recordset->Open(sql,m_pConnection.GetInterfacePtr(),adOpenStatic,adLockOptimistic,adCmdText);
recordset->put_CursorLocation(adUseClient);
}
catch(_com_error e)
{
AfxMessageBox((char *)e.Description());
return;
} */
// if(recordset->RecordCount==0)
// {
// continue;
// }
// AfxMessageBox("ko" );
// Name=(char*)(_bstr_t)recordset->GetCollect("name");
// Id=(char*)(_bstr_t)recordset->GetCollect("id");
// AfxMessageBox(Name);
curTime = CTime::GetCurrentTime();
tempStr.Format(("%d-%d-%d %d:%d:%d"), curTime.GetYear(),curTime.GetMonth(),
curTime.GetDay(),curTime.GetHour(),curTime.GetMinute(),curTime.GetSecond());
_bstr_t s="Insert into attendtmp(room,name,intime,id) values('"+RoomNum+"','"+Name+"','"+tempStr+"','"+Id+"')";
_variant_t affected;
try{
m_pConnection->Execute(s,&affected,adCmdText);
}
catch(_com_error e)
{
// AfxMessageBox((char *)e.Description());
return;
}
}
/* upscomport.SendData(senbuf,8);
Sleep(500);
if(upscomport.ReadData(recbuf,11)!=0)
{
CString r;
float ccc=recbuf[4]+recbuf[3]*pow(2,8);
// in3=atof(&recbuf[8])+atof(&recbuf[7])*pow(2,8);
ccc=ccc*220/10000;
r.Format("%f",ccc);
// AfxMessageBox(r);
// AfxMessageBox("UPStext!");
}
else{
// AfxMessageBox("ggggggg");
}
/*
BOOL bResult = TRUE;
DWORD Event = 0;
COMSTAT comstat;
DWORD dwError = 0;
DWORD CommEvent = 0;
bResult =WaitCommEvent(upscomport.m_hIDComDev, &Event, &(upscomport.m_OverlappedRead));
if(bResult)
{
AfxMessageBox("ggga");
bResult = ClearCommError(upscomport.m_hIDComDev, &dwError, &comstat);
if(comstat.cbInQue==0)
{
continue;
}
}
Event = WaitForMultipleObjects(2, upscomport.m_hEventArray, FALSE, 0);
switch (Event)
{
case 0: // read event
{
GetCommMask(upscomport.m_hIDComDev, &CommEvent);
if (CommEvent & EV_RXCHAR)
{
// Receive character event from port.
if(upscomport.ReadData(recbuf,16)!=0)
{
m=9;
result=0;
// if((buffer[0]==63)&&(buffer[15]==0))
// {
for(int i=1;i<10;i+=2)
{
for(int j=0;j<2;j++)
{
a[j]=buffer[i+j];
if(a[j]>=48&&a[j]<=57)
a[j]=a[j]-48;
else
if(a[j]>=0x41&&a[j]<=0x46)
a[j]=data[a[j]-0x41];
}
result=result+a[0]*pow(16,m--)+a[1]*pow(16,m--);
}
str.Format("%d",result);
str="0"+str;
break;
}
}
break;
}
case 1: // write event
{
// Write character event from port
//WriteChar();
break;
}
} // end swi
*/
Sleep(200);
}
// AfxMessageBox(str);
}
int CSerial::ReadData( void *buffer,int limit )
{
if( !m_bOpened || m_hIDComDev == NULL )
{
// char str[100];
// sprintf(str,"%d",m_bOpened);
// AfxMessageBox(str);
return( 0 );
}
BOOL bReadStatus;
DWORD dwBytesRead, dwErrorFlags;
COMSTAT ComStat;
ClearCommError( m_hIDComDev, &dwErrorFlags, &ComStat );
//PurgeComm(m_hIDComDev,PURGE_RXABORT|PURGE_RXCLEAR);
// CString str;
// str.Format("%d",ComStat.cbInQue);
if( !ComStat.cbInQue ) return( 0 );
/*CString str;
str.Format("%d",ComStat.cbInQue);
AfxMessageBox(str);
*/
dwBytesRead = (DWORD) ComStat.cbInQue;
if( limit < (int) dwBytesRead )
dwBytesRead = (DWORD) limit;
bReadStatus = ReadFile( m_hIDComDev, buffer, dwBytesRead, &dwBytesRead, &m_OverlappedRead );
PurgeComm(m_hIDComDev,PURGE_RXABORT|PURGE_RXCLEAR);
if( !bReadStatus )
{
if( GetLastError() == ERROR_IO_PENDING )
{
WaitForSingleObject( m_OverlappedRead.hEvent, 2000 );
return( (int) dwBytesRead );
}
return( 0 );
}
return dwBytesRead;
}