本月累计签到次数:

今天获取 积分

通讯

通讯

365 浏览

计算机与PLC的通讯格式及实现方法

电气控制类 柳草 2016-11-25 13:49 发表了文章 来自相关话题

计算机与PLC的通讯格式及实现方法
 可编程序控制器(PLC)都有一个编程口。以日本三菱公司生产的PLC为例(包括FX系列和A系列),其编程口为RS-422格式,根据PLC型号不同又分为8针座编程口和25针座编程口。对于后者,可直接将SC—08编程电缆将PLC的编程口和微型计算机的RS—232口连接起来;对于后者,则还需要一根转换电缆将PLC 的8针座编程口和25针编程电缆相连。无论何种情况,一旦将PLC用户程序由微型计算机编程环境传到PLC 用户程序区,其编程口大多就没有被再利用。其实,这是一种浪费。也就是说,可利用此编程口实现微型计算机和PLC 的数据通讯,将PLC的工作状态纳入微型计算机管理之下。
 
2 编程口操作命令类型与通讯端口初始化
 

 
串行通讯是计算机与其它机器之间进行通讯的一种常用方法,在Windows操作系统中提供了实现各种串行通讯的API函数。通过SC—08编程电缆或FX232AW模块,可将微型计算机的串行通讯口RS—232和PLC 的编程口连接起来,这样微型计算机就可对PLC的RAM区数据进行读、写操作。由PLC本身所具有的特性,可对PLC进行以下四种类型的操作:
 
(1)位元件或字元件状态读操作(CMD0);
 
(2)位元件或字元件状态写操作(CMD1);
 
(3)位元件强制ON操作(CMD7);
 
(4)位元件强制OFF操作(CMD8)。
 
另外,在进行上述四类操作以前,首先要对端口进行初始化操作,即设定通讯协议(包括设置通讯波特率、数据位数、数据停止位及奇偶校验)。在Windows的SDK中定义了一个结构DCB,该结构详细地说明了如何对通讯端口进行控制,所以通讯端口的初始化也是围绕着对这个结构的正确设置为中心进行的。用VC++语言实现端口初始化如下:
 
BOOL CSerial::Open(int nPort)
 
{
 
//nPort 为微型计算机串行通讯口端口号。nport=1为端口1;nPort=2为端口2。
 
char szPort\\[15\\];
 
DCB dob;
 
m_hIDComDev=CreateFile(szPort, GENERIC_READ│GENERIC_WRITE,O,NULL,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL│FILE_FLAG_OVERLAPPED,NULL);
 
dcb.DCB1ength=sizeof(DCB);
 
GetCommState(m_hIDComDev,&dcb);//取得通讯资源当前设置
 
dcb.BaudRate=9600;//设定波特率为9600
 
dcb.ByteSize=7;//7数据位
 
dcb.Parity=2;//偶校验
 
dcb.StopBits=0;//设定1个停止位
 
if(SetCommState(m_hIDComDev,&dcb)return(TRUE);
 
else return(FALSE);//设置端口,若设置成功则返回TRUE,否则返回FALSE
 

 
需要说明的是CSerial是一个用于串行通讯的类,它包含了进行串行通讯的所需的函数。除上述端口初始化成员函数Open外,还包括另两个重要成员函数:一个是endData,把数据从一个缓冲区发送到串行端口。另一个是ReadData,从端口的接收缓冲区中读入数据。
 
其次,在每进行一次上述四类操作中的一种操作以前,还要进行握手联络。对PLC发请求讯号ENQ(代码为OX05),然后读PLC 的响应讯号。如果读到的响应讯号为ACK(代码为OX06),则表示PLC已准备就绪,等待接收通讯数据。握手联络VC++语言示PLC已准备就绪,等待接收通讯数据。握手联络VC++语言实现为:
 
BOOL CNTJD1g::ReadFromPLC(char *Read_char char *Read_address,int Read_bytes)
 
{
 
CSerial Serial;//用于串行通讯的类
 
char read_BUFFER;
 
if(Serial.Open(2)//初始化串行口通讯口COM2
 
{ Serial.SendData(&ENQ_request,1);//发送联络讯号
 
Sleep(1000);//等待1秒钟
 
Serial.ReadData(&read_BUFFER,1);//读取PLC响应讯号
 
if(read_BUFFER==ACK)
 
{ 如果PLC响应讯号等于ACK,则进行上述四种操作:}}
 
Serial.Close()://操作完毕后,关闭通讯口
 
}
 
3 编程口命令操作
 

 
(1)位元件或字元件状态读操作
 
操作对象元件:PLC内部的X、Y、M、S、T、C、D元件;命令格式:
 
说明:①为读命令起始标志STX,代码为OX02;
 
②为位元件或字元件状态读命令CMDO,命令代码为OX30;
 
③为读位元件或字元件的4位起始地址,高位先发,低位后发,且是以ASCII码的形式发送;
 
④为一次读取位元件或字元件的个数,最多一次可读取OXff个字节的元件,以ASCII码的形式发送;
 
⑤为停止位标志ETX,代码为OX03;
 
⑥为2位和校验,和累计为②、③、④项代码,取其和最低两位转化成ASCII码,高位先发,低位后发。
 
在发送完上述命令格式代码后,就可直接读取PLC响应的信息。响应信息格式如下
 
VC++语言实现:
 
BOOL CNTJDlg::ReadFromPLC(char *Read_char char
 
*Read_address, int Read_bytes)
 
{
 
char senddatasum_CHECK\\[2\\];char readdatasum_CHECK\\[2\\]; char total_DATABYTES\\[2\\];
 
char readdatasum_check\\[2\\];int readdata_sum;
 
int datasum_check=0; int i;
 
Serial.SendData(&STX_start,1);/向PLC发送“开始”标志代码
 
Serial.SendData(&CMDO_read,1);//发送“读”命令代码datasum_check+=CMDO_read;
 
for(i=0;i<4;i++){Serial.SendData(&Read_address\\[i\\],1);//发送起始元件地址的ASCII代码datasum_check+=Read_address\\[i\\];}
 
Change to ASCII(total DATABYTES,Read_bytes);//将字节数转化成ASCII代码
 
for (i=0;i<2;i++){Serial.SendData(&total_DATABYTES\\[i\\],1);//发送元件字节数的ASCII代码)datasum_check+total_DATABYTES\\[i\\];}
 
Serial.SendData(&ETX_end,1);//发送“结束”标志代码senddatasum_CHECK+ETX_end;
 
Change_to_ASCII(senddatasum_CHECK,senddatasum_CHECK);//将“和”转化成ASCII码
 
for (i=0;i<2;i++) Serial.SendData(&senddatasum_CHECK\\[i\\],1);
 
Sleep(1000);//等待PLC响应
 
Serial.ReadData(&read_BUFFER,1);
 
if(read_BUFFER==STX_start){
 
readdata_sum=0;
 
for(i=0;i<2*Read_bytes;i++){Serial.ReadData(&Read_char\\[i\\],1);//读Read_bytes个字节readdata_sum+Read_char\\[i\\];}
 
Serial.ReadData(&read_BUFFER,1);
 
if(read_BUFFER==ETX_end){Serial.ReadData(readdatasum_CHECK,2);//读入的“和”的低2位ASCII码Readdata_sum+=ETX_end;}
 
Change_to_ASCII(readdatasum_check,readdata_sum);//将计算得到的“和”转化成ASCII码
 
if(*readdatasum_CHECK==*readdatasum_check)//“和”校验
 
{ AfxMessageBox(“数据读出成功!”)return TRUE;}
 
else { AfxMessageBox(“校验错误”)return FALSE.}
 
}
 
(2)位元件或字元件状态写操作
 
操作对象元件:同3(1);命令格式:
 
说明:①为写命令起始标志STX,代码为OX02;
 
②为位元件或字元件状态写命令CMD1,命令代码为OX31;
 
③为写位元件或字元件的4位起始地址,高位先发,低位后发,且是以ASCII码的形式发送;
 
④为一次写入位元件或字元件的个数,以ASCII码的形式发送;
 
⑤为待写到PLC RAM区的数据DATA,以ASCII码形式发送;
 
⑥为停止位标志ETX,代码为OX03;
 
⑦为2位和校验,和累计为②、③、④项代码,取其和最低两位转化成ASCII码,高位先发,低位后发。
 
VC++语言实现:
 
BOOL CNTJDlg::WritePLC(char *data_ADDRESS,char *Write_ASC,int bytesnumber)
 
{
 
char total_BYTES\\[2\\];char senddatasum_CHECK\\[2\\];
 
char read_BUFFER;char read_finishBUFFER;
 
int datasum_check=0; int i=0;
 
Serial.SendData(&STX_start,1);//向PLC发送“开始”标志代码
 
datasum_check=0;Serial.SendData(&CMD1_write,1);//发送“写”命令代码
 
datasum_check+CMD1_write;
 
for(i=0;i<4;i++) {Serial.SendData(&data_ADDRESS\\[i\\],1);//发送起始元件地址的ASCII码
 
datasum_check+=data_ADDRESS\\[i\\];
 
Change_to_ASCII(total_DATABYTES,bytesnumber);//将字节数转化成ASCII码
 
for(i=0;i<2;i++)
 
{
 
Serial.SendData(&total_BYTES\\,1);//发送元件字节数的ASCII代码
 
datasum_check+=total_BYTES\\[i\\];}
 
for {i=0;i {
 
Serial.SendData(&Write_ASC\\[i\\],1);//发送要写入的数据的ASCII码
 
datasum_check+=Write_ASC\\[i\\];}
 
Serial.SendData(&ETX_end,1);//发送“结束”标志代码
 
datasum_check+=ETX_end;
 
Change_to_ASCII(senddatasum_CHECK,datasum_check);//将“和”转化成ASCII码
 
Serial.SendData(&senddatasum_CHECK,2);
 
Sleep(1000); Serial.ReadData(&read_finishBUFFER,1);
 
if (read_finishBUFFER==ACK_reply)
 
{AfxMessageBox(“数据写入 OK”)return TRUE;}
 
else {AfxMessageBox(“数据写入失败”)return FALSE。}
 
(3)位元件强制ON操作
 
操作对象:X、Y、M、S、T、C元件;
 
命令格式:
 
说明:①为强制OFF命令起始标志STX,代码为OX02;
 
②为强制OFF命令CMD8,命令代码为OX38H;
 
③为强制OFF位元件4位起始地址,高位先发,低位后发,以ASCII码形式发送;
 
④为停止位标志ETX,代码为OX03;
 
⑤为2位和校验,和累计为②、③、④项代码,取其和最低两位转化成ASCII码,高位先发,低位后发。
 
VC++语言实现:
 
void NTJDlg::ForceOffOperation (char *OFF_Address)
 
{
 
int i;
 
char Sum_Check\\[2\\];
 
char read_buffer;
 
int Sum=0;
 
Serial.SendData(&STX_start,1);//向PLC发送“开始”标志代码
 
Serial.SendData(&CMD8_ForceOFF,1);//发送“OFF”命令代码
 
Sum=CMD8_ForceOFF;
 
for (i=0;i<4;i++) {
 
Serial.SendData(&OFF_Address\\[i\\],1);//发送起始元件地址的ASCII码
 
Sum+=OFF_Address\\[i\\];}
 
Serial.SendData(&ETX_end,1);//发送“结束”标志代码
 
Sum+=ETX_end;
 
Change_to_ASCII(Sum_Check,Sum);//将“和”转化成ASCII码
 
Serial.SendData(&Sum_Check,2);
 
Skeeo(1000);
 
Serial.ReadData(&read_buffer,1);
 
if(read_fininhBUFFER==ACK_reply) AfxMessageBox(“OFF 操作 OK ”);
 
else AfxMessageBox(“OFF 操作失败”)。
 

 
注意:必须严格按照上述四种操作命令格式进行发送,在发送前,起始地址、数据、数据个数、校验和都必须按位转换成ASCII码。从PLC读到的数据亦是ASCII码形式,需要经过适当转换才能利用。另外,要注意强制命令地址与读写地址的顺序不是一样,且一次最多只能传送64个字节数据。
 
4 结论
 

 
利用上述四种操作命令,就可对PLC的RAM区数据进行管理操作。将PLC的工作状态纳入微型计算机管理之下。在此基础上,用户可以应用VC很方便地设计自己的PLC人机接口界面,为监控与管理PLC的运行提供一种良好的方法。 查看全部
计算机与PLC的通讯格式及实现方法
 可编程序控制器(PLC)都有一个编程口。以日本三菱公司生产的PLC为例(包括FX系列和A系列),其编程口为RS-422格式,根据PLC型号不同又分为8针座编程口和25针座编程口。对于后者,可直接将SC—08编程电缆将PLC的编程口和微型计算机的RS—232口连接起来;对于后者,则还需要一根转换电缆将PLC 的8针座编程口和25针编程电缆相连。无论何种情况,一旦将PLC用户程序由微型计算机编程环境传到PLC 用户程序区,其编程口大多就没有被再利用。其实,这是一种浪费。也就是说,可利用此编程口实现微型计算机和PLC 的数据通讯,将PLC的工作状态纳入微型计算机管理之下。
 
2 编程口操作命令类型与通讯端口初始化
 

 
串行通讯是计算机与其它机器之间进行通讯的一种常用方法,在Windows操作系统中提供了实现各种串行通讯的API函数。通过SC—08编程电缆或FX232AW模块,可将微型计算机的串行通讯口RS—232和PLC 的编程口连接起来,这样微型计算机就可对PLC的RAM区数据进行读、写操作。由PLC本身所具有的特性,可对PLC进行以下四种类型的操作:
 
(1)位元件或字元件状态读操作(CMD0);
 
(2)位元件或字元件状态写操作(CMD1);
 
(3)位元件强制ON操作(CMD7);
 
(4)位元件强制OFF操作(CMD8)。
 
另外,在进行上述四类操作以前,首先要对端口进行初始化操作,即设定通讯协议(包括设置通讯波特率、数据位数、数据停止位及奇偶校验)。在Windows的SDK中定义了一个结构DCB,该结构详细地说明了如何对通讯端口进行控制,所以通讯端口的初始化也是围绕着对这个结构的正确设置为中心进行的。用VC++语言实现端口初始化如下:
 
BOOL CSerial::Open(int nPort)
 
{
 
//nPort 为微型计算机串行通讯口端口号。nport=1为端口1;nPort=2为端口2。
 
char szPort\\[15\\];
 
DCB dob;
 
m_hIDComDev=CreateFile(szPort, GENERIC_READ│GENERIC_WRITE,O,NULL,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL│FILE_FLAG_OVERLAPPED,NULL);
 
dcb.DCB1ength=sizeof(DCB);
 
GetCommState(m_hIDComDev,&dcb);//取得通讯资源当前设置
 
dcb.BaudRate=9600;//设定波特率为9600
 
dcb.ByteSize=7;//7数据位
 
dcb.Parity=2;//偶校验
 
dcb.StopBits=0;//设定1个停止位
 
if(SetCommState(m_hIDComDev,&dcb)return(TRUE);
 
else return(FALSE);//设置端口,若设置成功则返回TRUE,否则返回FALSE
 

 
需要说明的是CSerial是一个用于串行通讯的类,它包含了进行串行通讯的所需的函数。除上述端口初始化成员函数Open外,还包括另两个重要成员函数:一个是endData,把数据从一个缓冲区发送到串行端口。另一个是ReadData,从端口的接收缓冲区中读入数据。
 
其次,在每进行一次上述四类操作中的一种操作以前,还要进行握手联络。对PLC发请求讯号ENQ(代码为OX05),然后读PLC 的响应讯号。如果读到的响应讯号为ACK(代码为OX06),则表示PLC已准备就绪,等待接收通讯数据。握手联络VC++语言示PLC已准备就绪,等待接收通讯数据。握手联络VC++语言实现为:
 
BOOL CNTJD1g::ReadFromPLC(char *Read_char char *Read_address,int Read_bytes)
 
{
 
CSerial Serial;//用于串行通讯的类
 
char read_BUFFER;
 
if(Serial.Open(2)//初始化串行口通讯口COM2
 
{ Serial.SendData(&ENQ_request,1);//发送联络讯号
 
Sleep(1000);//等待1秒钟
 
Serial.ReadData(&read_BUFFER,1);//读取PLC响应讯号
 
if(read_BUFFER==ACK)
 
{ 如果PLC响应讯号等于ACK,则进行上述四种操作:}}
 
Serial.Close()://操作完毕后,关闭通讯口
 
}
 
3 编程口命令操作
 

 
(1)位元件或字元件状态读操作
 
操作对象元件:PLC内部的X、Y、M、S、T、C、D元件;命令格式:
 
说明:①为读命令起始标志STX,代码为OX02;
 
②为位元件或字元件状态读命令CMDO,命令代码为OX30;
 
③为读位元件或字元件的4位起始地址,高位先发,低位后发,且是以ASCII码的形式发送;
 
④为一次读取位元件或字元件的个数,最多一次可读取OXff个字节的元件,以ASCII码的形式发送;
 
⑤为停止位标志ETX,代码为OX03;
 
⑥为2位和校验,和累计为②、③、④项代码,取其和最低两位转化成ASCII码,高位先发,低位后发。
 
在发送完上述命令格式代码后,就可直接读取PLC响应的信息。响应信息格式如下
 
VC++语言实现:
 
BOOL CNTJDlg::ReadFromPLC(char *Read_char char
 
*Read_address, int Read_bytes)
 
{
 
char senddatasum_CHECK\\[2\\];char readdatasum_CHECK\\[2\\]; char total_DATABYTES\\[2\\];
 
char readdatasum_check\\[2\\];int readdata_sum;
 
int datasum_check=0; int i;
 
Serial.SendData(&STX_start,1);/向PLC发送“开始”标志代码
 
Serial.SendData(&CMDO_read,1);//发送“读”命令代码datasum_check+=CMDO_read;
 
for(i=0;i<4;i++){Serial.SendData(&Read_address\\[i\\],1);//发送起始元件地址的ASCII代码datasum_check+=Read_address\\[i\\];}
 
Change to ASCII(total DATABYTES,Read_bytes);//将字节数转化成ASCII代码
 
for (i=0;i<2;i++){Serial.SendData(&total_DATABYTES\\[i\\],1);//发送元件字节数的ASCII代码)datasum_check+total_DATABYTES\\[i\\];}
 
Serial.SendData(&ETX_end,1);//发送“结束”标志代码senddatasum_CHECK+ETX_end;
 
Change_to_ASCII(senddatasum_CHECK,senddatasum_CHECK);//将“和”转化成ASCII码
 
for (i=0;i<2;i++) Serial.SendData(&senddatasum_CHECK\\[i\\],1);
 
Sleep(1000);//等待PLC响应
 
Serial.ReadData(&read_BUFFER,1);
 
if(read_BUFFER==STX_start){
 
readdata_sum=0;
 
for(i=0;i<2*Read_bytes;i++){Serial.ReadData(&Read_char\\[i\\],1);//读Read_bytes个字节readdata_sum+Read_char\\[i\\];}
 
Serial.ReadData(&read_BUFFER,1);
 
if(read_BUFFER==ETX_end){Serial.ReadData(readdatasum_CHECK,2);//读入的“和”的低2位ASCII码Readdata_sum+=ETX_end;}
 
Change_to_ASCII(readdatasum_check,readdata_sum);//将计算得到的“和”转化成ASCII码
 
if(*readdatasum_CHECK==*readdatasum_check)//“和”校验
 
{ AfxMessageBox(“数据读出成功!”)return TRUE;}
 
else { AfxMessageBox(“校验错误”)return FALSE.}
 
}
 
(2)位元件或字元件状态写操作
 
操作对象元件:同3(1);命令格式:
 
说明:①为写命令起始标志STX,代码为OX02;
 
②为位元件或字元件状态写命令CMD1,命令代码为OX31;
 
③为写位元件或字元件的4位起始地址,高位先发,低位后发,且是以ASCII码的形式发送;
 
④为一次写入位元件或字元件的个数,以ASCII码的形式发送;
 
⑤为待写到PLC RAM区的数据DATA,以ASCII码形式发送;
 
⑥为停止位标志ETX,代码为OX03;
 
⑦为2位和校验,和累计为②、③、④项代码,取其和最低两位转化成ASCII码,高位先发,低位后发。
 
VC++语言实现:
 
BOOL CNTJDlg::WritePLC(char *data_ADDRESS,char *Write_ASC,int bytesnumber)
 
{
 
char total_BYTES\\[2\\];char senddatasum_CHECK\\[2\\];
 
char read_BUFFER;char read_finishBUFFER;
 
int datasum_check=0; int i=0;
 
Serial.SendData(&STX_start,1);//向PLC发送“开始”标志代码
 
datasum_check=0;Serial.SendData(&CMD1_write,1);//发送“写”命令代码
 
datasum_check+CMD1_write;
 
for(i=0;i<4;i++) {Serial.SendData(&data_ADDRESS\\[i\\],1);//发送起始元件地址的ASCII码
 
datasum_check+=data_ADDRESS\\[i\\];
 
Change_to_ASCII(total_DATABYTES,bytesnumber);//将字节数转化成ASCII码
 
for(i=0;i<2;i++)
 
{
 
Serial.SendData(&total_BYTES\\,1);//发送元件字节数的ASCII代码
 
datasum_check+=total_BYTES\\[i\\];}
 
for {i=0;i {
 
Serial.SendData(&Write_ASC\\[i\\],1);//发送要写入的数据的ASCII码
 
datasum_check+=Write_ASC\\[i\\];}
 
Serial.SendData(&ETX_end,1);//发送“结束”标志代码
 
datasum_check+=ETX_end;
 
Change_to_ASCII(senddatasum_CHECK,datasum_check);//将“和”转化成ASCII码
 
Serial.SendData(&senddatasum_CHECK,2);
 
Sleep(1000); Serial.ReadData(&read_finishBUFFER,1);
 
if (read_finishBUFFER==ACK_reply)
 
{AfxMessageBox(“数据写入 OK”)return TRUE;}
 
else {AfxMessageBox(“数据写入失败”)return FALSE。}
 
(3)位元件强制ON操作
 
操作对象:X、Y、M、S、T、C元件;
 
命令格式:
 
说明:①为强制OFF命令起始标志STX,代码为OX02;
 
②为强制OFF命令CMD8,命令代码为OX38H;
 
③为强制OFF位元件4位起始地址,高位先发,低位后发,以ASCII码形式发送;
 
④为停止位标志ETX,代码为OX03;
 
⑤为2位和校验,和累计为②、③、④项代码,取其和最低两位转化成ASCII码,高位先发,低位后发。
 
VC++语言实现:
 
void NTJDlg::ForceOffOperation (char *OFF_Address)
 
{
 
int i;
 
char Sum_Check\\[2\\];
 
char read_buffer;
 
int Sum=0;
 
Serial.SendData(&STX_start,1);//向PLC发送“开始”标志代码
 
Serial.SendData(&CMD8_ForceOFF,1);//发送“OFF”命令代码
 
Sum=CMD8_ForceOFF;
 
for (i=0;i<4;i++) {
 
Serial.SendData(&OFF_Address\\[i\\],1);//发送起始元件地址的ASCII码
 
Sum+=OFF_Address\\[i\\];}
 
Serial.SendData(&ETX_end,1);//发送“结束”标志代码
 
Sum+=ETX_end;
 
Change_to_ASCII(Sum_Check,Sum);//将“和”转化成ASCII码
 
Serial.SendData(&Sum_Check,2);
 
Skeeo(1000);
 
Serial.ReadData(&read_buffer,1);
 
if(read_fininhBUFFER==ACK_reply) AfxMessageBox(“OFF 操作 OK ”);
 
else AfxMessageBox(“OFF 操作失败”)。
 

 
注意:必须严格按照上述四种操作命令格式进行发送,在发送前,起始地址、数据、数据个数、校验和都必须按位转换成ASCII码。从PLC读到的数据亦是ASCII码形式,需要经过适当转换才能利用。另外,要注意强制命令地址与读写地址的顺序不是一样,且一次最多只能传送64个字节数据。
 
4 结论
 

 
利用上述四种操作命令,就可对PLC的RAM区数据进行管理操作。将PLC的工作状态纳入微型计算机管理之下。在此基础上,用户可以应用VC很方便地设计自己的PLC人机接口界面,为监控与管理PLC的运行提供一种良好的方法。
5 回答
8 回答

空间激光通信?什么新技术?

机械自动化类 发财大计 2016-11-21 10:01 回复了问题 • 9 人关注 来自相关话题

8 回答

什么标准能满足智能制造的通信要求?

电气控制类 entschuldig 2016-10-21 10:31 回复了问题 • 9 人关注 来自相关话题

条新动态, 点击查看
仲夏之夜

仲夏之夜 回答了问题 • 2016-10-21 10:25 • 8 个回复 不感兴趣

什么标准能满足智能制造的通信要求?

赞同来自:

智慧制造要求打破多层递阶的信息流

智慧制造系统有三类通信:B2B、B2M和M2M。B2B和B2M属于软实时通信其时间响应在毫秒级至几分钟之间。M2M属于硬实时通信,时间相应在微秒至毫秒之间。德国工业4.0的WG2执行委员会精确定义了三类通信事务:
1、B2B... 显示全部 »
智慧制造要求打破多层递阶的信息流

智慧制造系统有三类通信:B2B、B2M和M2M。B2B和B2M属于软实时通信其时间响应在毫秒级至几分钟之间。M2M属于硬实时通信,时间相应在微秒至毫秒之间。德国工业4.0的WG2执行委员会精确定义了三类通信事务:
1、B2B通信两类业务过程相互通信。信息交换所需时间由几个毫秒到几分钟。例如:ERP应用与MES应用的信息交换;HMI和MES见的信息交换;MES与另一MES的信息交换;或者传感器与云的交换。
2、B2M通信软实时过程与硬实时过程通信。所需的交换时间由几毫秒到几分钟。例如:业务应用过程与机械装备间的信息交换HMI与PLC,MES与PLC的信息交换。
3、M2M通信在自动化环境下两个过程的通信,这些过程可以是硬实时过程与硬实时过程的通信,也可以是软实时过程与硬实时过程的通信。例如:一个机器人平台控制器与一个手持式机器人控制器之间的横向通信。
其交换信息的时间由微秒至几个毫秒,必须发生在硬实时、且为确定性的循环内。也可以把确定性看成是具有一定时间要求的QoS(服务质量),即在一个有保证的时间间隔(如100微妙)完成响应。另一个例子是两个控制器之间的横向通信(软实时的快速、循环、独立于现场总线的通信)。已经实现运用无线移动通信完成M2M,这里是指一个设备的接口(即装载设备上的SIM卡)通过移动通信与某个IT过程进行通信。
 
中国拿下5G短码标准的技术价值和商业意义

 据了解,由于短码的讨论分为控制信道和数据信道,各公司从性能、实现复杂度,以及可行性等角度对几个候选编码进行了全面的分析。有媒体报道称,会议的讨论异常激烈,几乎所有的公司都参与其中,在5G控制信道eMBB场景编码方案... 显示全部 »
中国拿下5G短码标准的技术价值和商业意义

 据了解,由于短码的讨论分为控制信道和数据信道,各公司从性能、实现复杂度,以及可行性等角度对几个候选编码进行了全面的分析。有媒体报道称,会议的讨论异常激烈,几乎所有的公司都参与其中,在5G控制信道eMBB场景编码方案上,以华为主导的中国队对抗美欧联军,最后还拉了51家公司同签,力压LDPC的37家公司,真是一场来之不易、令人欣喜的胜利。数十年来,通信行业里各家公司的实力此消彼长;中国公司从一开始被圈在场外,到逐渐开始参与标准制定,到主导部分条目的立项,再到今天这样全方位出击,确实反映了国内公司的技术进步和商业地位。
5 回答
8 回答

空间激光通信?什么新技术?

机械自动化类 发财大计 2016-11-21 10:01 回复了问题 • 9 人关注 来自相关话题

8 回答

什么标准能满足智能制造的通信要求?

电气控制类 entschuldig 2016-10-21 10:31 回复了问题 • 9 人关注 来自相关话题

365 浏览

计算机与PLC的通讯格式及实现方法

电气控制类 柳草 2016-11-25 13:49 发表了文章 来自相关话题

计算机与PLC的通讯格式及实现方法
 可编程序控制器(PLC)都有一个编程口。以日本三菱公司生产的PLC为例(包括FX系列和A系列),其编程口为RS-422格式,根据PLC型号不同又分为8针座编程口和25针座编程口。对于后者,可直接将SC—08编程电缆将PLC的编程口和微型计算机的RS—232口连接起来;对于后者,则还需要一根转换电缆将PLC 的8针座编程口和25针编程电缆相连。无论何种情况,一旦将PLC用户程序由微型计算机编程环境传到PLC 用户程序区,其编程口大多就没有被再利用。其实,这是一种浪费。也就是说,可利用此编程口实现微型计算机和PLC 的数据通讯,将PLC的工作状态纳入微型计算机管理之下。
 
2 编程口操作命令类型与通讯端口初始化
 

 
串行通讯是计算机与其它机器之间进行通讯的一种常用方法,在Windows操作系统中提供了实现各种串行通讯的API函数。通过SC—08编程电缆或FX232AW模块,可将微型计算机的串行通讯口RS—232和PLC 的编程口连接起来,这样微型计算机就可对PLC的RAM区数据进行读、写操作。由PLC本身所具有的特性,可对PLC进行以下四种类型的操作:
 
(1)位元件或字元件状态读操作(CMD0);
 
(2)位元件或字元件状态写操作(CMD1);
 
(3)位元件强制ON操作(CMD7);
 
(4)位元件强制OFF操作(CMD8)。
 
另外,在进行上述四类操作以前,首先要对端口进行初始化操作,即设定通讯协议(包括设置通讯波特率、数据位数、数据停止位及奇偶校验)。在Windows的SDK中定义了一个结构DCB,该结构详细地说明了如何对通讯端口进行控制,所以通讯端口的初始化也是围绕着对这个结构的正确设置为中心进行的。用VC++语言实现端口初始化如下:
 
BOOL CSerial::Open(int nPort)
 
{
 
//nPort 为微型计算机串行通讯口端口号。nport=1为端口1;nPort=2为端口2。
 
char szPort\\[15\\];
 
DCB dob;
 
m_hIDComDev=CreateFile(szPort, GENERIC_READ│GENERIC_WRITE,O,NULL,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL│FILE_FLAG_OVERLAPPED,NULL);
 
dcb.DCB1ength=sizeof(DCB);
 
GetCommState(m_hIDComDev,&dcb);//取得通讯资源当前设置
 
dcb.BaudRate=9600;//设定波特率为9600
 
dcb.ByteSize=7;//7数据位
 
dcb.Parity=2;//偶校验
 
dcb.StopBits=0;//设定1个停止位
 
if(SetCommState(m_hIDComDev,&dcb)return(TRUE);
 
else return(FALSE);//设置端口,若设置成功则返回TRUE,否则返回FALSE
 

 
需要说明的是CSerial是一个用于串行通讯的类,它包含了进行串行通讯的所需的函数。除上述端口初始化成员函数Open外,还包括另两个重要成员函数:一个是endData,把数据从一个缓冲区发送到串行端口。另一个是ReadData,从端口的接收缓冲区中读入数据。
 
其次,在每进行一次上述四类操作中的一种操作以前,还要进行握手联络。对PLC发请求讯号ENQ(代码为OX05),然后读PLC 的响应讯号。如果读到的响应讯号为ACK(代码为OX06),则表示PLC已准备就绪,等待接收通讯数据。握手联络VC++语言示PLC已准备就绪,等待接收通讯数据。握手联络VC++语言实现为:
 
BOOL CNTJD1g::ReadFromPLC(char *Read_char char *Read_address,int Read_bytes)
 
{
 
CSerial Serial;//用于串行通讯的类
 
char read_BUFFER;
 
if(Serial.Open(2)//初始化串行口通讯口COM2
 
{ Serial.SendData(&ENQ_request,1);//发送联络讯号
 
Sleep(1000);//等待1秒钟
 
Serial.ReadData(&read_BUFFER,1);//读取PLC响应讯号
 
if(read_BUFFER==ACK)
 
{ 如果PLC响应讯号等于ACK,则进行上述四种操作:}}
 
Serial.Close()://操作完毕后,关闭通讯口
 
}
 
3 编程口命令操作
 

 
(1)位元件或字元件状态读操作
 
操作对象元件:PLC内部的X、Y、M、S、T、C、D元件;命令格式:
 
说明:①为读命令起始标志STX,代码为OX02;
 
②为位元件或字元件状态读命令CMDO,命令代码为OX30;
 
③为读位元件或字元件的4位起始地址,高位先发,低位后发,且是以ASCII码的形式发送;
 
④为一次读取位元件或字元件的个数,最多一次可读取OXff个字节的元件,以ASCII码的形式发送;
 
⑤为停止位标志ETX,代码为OX03;
 
⑥为2位和校验,和累计为②、③、④项代码,取其和最低两位转化成ASCII码,高位先发,低位后发。
 
在发送完上述命令格式代码后,就可直接读取PLC响应的信息。响应信息格式如下
 
VC++语言实现:
 
BOOL CNTJDlg::ReadFromPLC(char *Read_char char
 
*Read_address, int Read_bytes)
 
{
 
char senddatasum_CHECK\\[2\\];char readdatasum_CHECK\\[2\\]; char total_DATABYTES\\[2\\];
 
char readdatasum_check\\[2\\];int readdata_sum;
 
int datasum_check=0; int i;
 
Serial.SendData(&STX_start,1);/向PLC发送“开始”标志代码
 
Serial.SendData(&CMDO_read,1);//发送“读”命令代码datasum_check+=CMDO_read;
 
for(i=0;i<4;i++){Serial.SendData(&Read_address\\[i\\],1);//发送起始元件地址的ASCII代码datasum_check+=Read_address\\[i\\];}
 
Change to ASCII(total DATABYTES,Read_bytes);//将字节数转化成ASCII代码
 
for (i=0;i<2;i++){Serial.SendData(&total_DATABYTES\\[i\\],1);//发送元件字节数的ASCII代码)datasum_check+total_DATABYTES\\[i\\];}
 
Serial.SendData(&ETX_end,1);//发送“结束”标志代码senddatasum_CHECK+ETX_end;
 
Change_to_ASCII(senddatasum_CHECK,senddatasum_CHECK);//将“和”转化成ASCII码
 
for (i=0;i<2;i++) Serial.SendData(&senddatasum_CHECK\\[i\\],1);
 
Sleep(1000);//等待PLC响应
 
Serial.ReadData(&read_BUFFER,1);
 
if(read_BUFFER==STX_start){
 
readdata_sum=0;
 
for(i=0;i<2*Read_bytes;i++){Serial.ReadData(&Read_char\\[i\\],1);//读Read_bytes个字节readdata_sum+Read_char\\[i\\];}
 
Serial.ReadData(&read_BUFFER,1);
 
if(read_BUFFER==ETX_end){Serial.ReadData(readdatasum_CHECK,2);//读入的“和”的低2位ASCII码Readdata_sum+=ETX_end;}
 
Change_to_ASCII(readdatasum_check,readdata_sum);//将计算得到的“和”转化成ASCII码
 
if(*readdatasum_CHECK==*readdatasum_check)//“和”校验
 
{ AfxMessageBox(“数据读出成功!”)return TRUE;}
 
else { AfxMessageBox(“校验错误”)return FALSE.}
 
}
 
(2)位元件或字元件状态写操作
 
操作对象元件:同3(1);命令格式:
 
说明:①为写命令起始标志STX,代码为OX02;
 
②为位元件或字元件状态写命令CMD1,命令代码为OX31;
 
③为写位元件或字元件的4位起始地址,高位先发,低位后发,且是以ASCII码的形式发送;
 
④为一次写入位元件或字元件的个数,以ASCII码的形式发送;
 
⑤为待写到PLC RAM区的数据DATA,以ASCII码形式发送;
 
⑥为停止位标志ETX,代码为OX03;
 
⑦为2位和校验,和累计为②、③、④项代码,取其和最低两位转化成ASCII码,高位先发,低位后发。
 
VC++语言实现:
 
BOOL CNTJDlg::WritePLC(char *data_ADDRESS,char *Write_ASC,int bytesnumber)
 
{
 
char total_BYTES\\[2\\];char senddatasum_CHECK\\[2\\];
 
char read_BUFFER;char read_finishBUFFER;
 
int datasum_check=0; int i=0;
 
Serial.SendData(&STX_start,1);//向PLC发送“开始”标志代码
 
datasum_check=0;Serial.SendData(&CMD1_write,1);//发送“写”命令代码
 
datasum_check+CMD1_write;
 
for(i=0;i<4;i++) {Serial.SendData(&data_ADDRESS\\[i\\],1);//发送起始元件地址的ASCII码
 
datasum_check+=data_ADDRESS\\[i\\];
 
Change_to_ASCII(total_DATABYTES,bytesnumber);//将字节数转化成ASCII码
 
for(i=0;i<2;i++)
 
{
 
Serial.SendData(&total_BYTES\\,1);//发送元件字节数的ASCII代码
 
datasum_check+=total_BYTES\\[i\\];}
 
for {i=0;i {
 
Serial.SendData(&Write_ASC\\[i\\],1);//发送要写入的数据的ASCII码
 
datasum_check+=Write_ASC\\[i\\];}
 
Serial.SendData(&ETX_end,1);//发送“结束”标志代码
 
datasum_check+=ETX_end;
 
Change_to_ASCII(senddatasum_CHECK,datasum_check);//将“和”转化成ASCII码
 
Serial.SendData(&senddatasum_CHECK,2);
 
Sleep(1000); Serial.ReadData(&read_finishBUFFER,1);
 
if (read_finishBUFFER==ACK_reply)
 
{AfxMessageBox(“数据写入 OK”)return TRUE;}
 
else {AfxMessageBox(“数据写入失败”)return FALSE。}
 
(3)位元件强制ON操作
 
操作对象:X、Y、M、S、T、C元件;
 
命令格式:
 
说明:①为强制OFF命令起始标志STX,代码为OX02;
 
②为强制OFF命令CMD8,命令代码为OX38H;
 
③为强制OFF位元件4位起始地址,高位先发,低位后发,以ASCII码形式发送;
 
④为停止位标志ETX,代码为OX03;
 
⑤为2位和校验,和累计为②、③、④项代码,取其和最低两位转化成ASCII码,高位先发,低位后发。
 
VC++语言实现:
 
void NTJDlg::ForceOffOperation (char *OFF_Address)
 
{
 
int i;
 
char Sum_Check\\[2\\];
 
char read_buffer;
 
int Sum=0;
 
Serial.SendData(&STX_start,1);//向PLC发送“开始”标志代码
 
Serial.SendData(&CMD8_ForceOFF,1);//发送“OFF”命令代码
 
Sum=CMD8_ForceOFF;
 
for (i=0;i<4;i++) {
 
Serial.SendData(&OFF_Address\\[i\\],1);//发送起始元件地址的ASCII码
 
Sum+=OFF_Address\\[i\\];}
 
Serial.SendData(&ETX_end,1);//发送“结束”标志代码
 
Sum+=ETX_end;
 
Change_to_ASCII(Sum_Check,Sum);//将“和”转化成ASCII码
 
Serial.SendData(&Sum_Check,2);
 
Skeeo(1000);
 
Serial.ReadData(&read_buffer,1);
 
if(read_fininhBUFFER==ACK_reply) AfxMessageBox(“OFF 操作 OK ”);
 
else AfxMessageBox(“OFF 操作失败”)。
 

 
注意:必须严格按照上述四种操作命令格式进行发送,在发送前,起始地址、数据、数据个数、校验和都必须按位转换成ASCII码。从PLC读到的数据亦是ASCII码形式,需要经过适当转换才能利用。另外,要注意强制命令地址与读写地址的顺序不是一样,且一次最多只能传送64个字节数据。
 
4 结论
 

 
利用上述四种操作命令,就可对PLC的RAM区数据进行管理操作。将PLC的工作状态纳入微型计算机管理之下。在此基础上,用户可以应用VC很方便地设计自己的PLC人机接口界面,为监控与管理PLC的运行提供一种良好的方法。 查看全部
计算机与PLC的通讯格式及实现方法
 可编程序控制器(PLC)都有一个编程口。以日本三菱公司生产的PLC为例(包括FX系列和A系列),其编程口为RS-422格式,根据PLC型号不同又分为8针座编程口和25针座编程口。对于后者,可直接将SC—08编程电缆将PLC的编程口和微型计算机的RS—232口连接起来;对于后者,则还需要一根转换电缆将PLC 的8针座编程口和25针编程电缆相连。无论何种情况,一旦将PLC用户程序由微型计算机编程环境传到PLC 用户程序区,其编程口大多就没有被再利用。其实,这是一种浪费。也就是说,可利用此编程口实现微型计算机和PLC 的数据通讯,将PLC的工作状态纳入微型计算机管理之下。
 
2 编程口操作命令类型与通讯端口初始化
 

 
串行通讯是计算机与其它机器之间进行通讯的一种常用方法,在Windows操作系统中提供了实现各种串行通讯的API函数。通过SC—08编程电缆或FX232AW模块,可将微型计算机的串行通讯口RS—232和PLC 的编程口连接起来,这样微型计算机就可对PLC的RAM区数据进行读、写操作。由PLC本身所具有的特性,可对PLC进行以下四种类型的操作:
 
(1)位元件或字元件状态读操作(CMD0);
 
(2)位元件或字元件状态写操作(CMD1);
 
(3)位元件强制ON操作(CMD7);
 
(4)位元件强制OFF操作(CMD8)。
 
另外,在进行上述四类操作以前,首先要对端口进行初始化操作,即设定通讯协议(包括设置通讯波特率、数据位数、数据停止位及奇偶校验)。在Windows的SDK中定义了一个结构DCB,该结构详细地说明了如何对通讯端口进行控制,所以通讯端口的初始化也是围绕着对这个结构的正确设置为中心进行的。用VC++语言实现端口初始化如下:
 
BOOL CSerial::Open(int nPort)
 
{
 
//nPort 为微型计算机串行通讯口端口号。nport=1为端口1;nPort=2为端口2。
 
char szPort\\[15\\];
 
DCB dob;
 
m_hIDComDev=CreateFile(szPort, GENERIC_READ│GENERIC_WRITE,O,NULL,OPEN_EXISTING,FILE_ATTRIBUTE_NORMAL│FILE_FLAG_OVERLAPPED,NULL);
 
dcb.DCB1ength=sizeof(DCB);
 
GetCommState(m_hIDComDev,&dcb);//取得通讯资源当前设置
 
dcb.BaudRate=9600;//设定波特率为9600
 
dcb.ByteSize=7;//7数据位
 
dcb.Parity=2;//偶校验
 
dcb.StopBits=0;//设定1个停止位
 
if(SetCommState(m_hIDComDev,&dcb)return(TRUE);
 
else return(FALSE);//设置端口,若设置成功则返回TRUE,否则返回FALSE
 

 
需要说明的是CSerial是一个用于串行通讯的类,它包含了进行串行通讯的所需的函数。除上述端口初始化成员函数Open外,还包括另两个重要成员函数:一个是endData,把数据从一个缓冲区发送到串行端口。另一个是ReadData,从端口的接收缓冲区中读入数据。
 
其次,在每进行一次上述四类操作中的一种操作以前,还要进行握手联络。对PLC发请求讯号ENQ(代码为OX05),然后读PLC 的响应讯号。如果读到的响应讯号为ACK(代码为OX06),则表示PLC已准备就绪,等待接收通讯数据。握手联络VC++语言示PLC已准备就绪,等待接收通讯数据。握手联络VC++语言实现为:
 
BOOL CNTJD1g::ReadFromPLC(char *Read_char char *Read_address,int Read_bytes)
 
{
 
CSerial Serial;//用于串行通讯的类
 
char read_BUFFER;
 
if(Serial.Open(2)//初始化串行口通讯口COM2
 
{ Serial.SendData(&ENQ_request,1);//发送联络讯号
 
Sleep(1000);//等待1秒钟
 
Serial.ReadData(&read_BUFFER,1);//读取PLC响应讯号
 
if(read_BUFFER==ACK)
 
{ 如果PLC响应讯号等于ACK,则进行上述四种操作:}}
 
Serial.Close()://操作完毕后,关闭通讯口
 
}
 
3 编程口命令操作
 

 
(1)位元件或字元件状态读操作
 
操作对象元件:PLC内部的X、Y、M、S、T、C、D元件;命令格式:
 
说明:①为读命令起始标志STX,代码为OX02;
 
②为位元件或字元件状态读命令CMDO,命令代码为OX30;
 
③为读位元件或字元件的4位起始地址,高位先发,低位后发,且是以ASCII码的形式发送;
 
④为一次读取位元件或字元件的个数,最多一次可读取OXff个字节的元件,以ASCII码的形式发送;
 
⑤为停止位标志ETX,代码为OX03;
 
⑥为2位和校验,和累计为②、③、④项代码,取其和最低两位转化成ASCII码,高位先发,低位后发。
 
在发送完上述命令格式代码后,就可直接读取PLC响应的信息。响应信息格式如下
 
VC++语言实现:
 
BOOL CNTJDlg::ReadFromPLC(char *Read_char char
 
*Read_address, int Read_bytes)
 
{
 
char senddatasum_CHECK\\[2\\];char readdatasum_CHECK\\[2\\]; char total_DATABYTES\\[2\\];
 
char readdatasum_check\\[2\\];int readdata_sum;
 
int datasum_check=0; int i;
 
Serial.SendData(&STX_start,1);/向PLC发送“开始”标志代码
 
Serial.SendData(&CMDO_read,1);//发送“读”命令代码datasum_check+=CMDO_read;
 
for(i=0;i<4;i++){Serial.SendData(&Read_address\\[i\\],1);//发送起始元件地址的ASCII代码datasum_check+=Read_address\\[i\\];}
 
Change to ASCII(total DATABYTES,Read_bytes);//将字节数转化成ASCII代码
 
for (i=0;i<2;i++){Serial.SendData(&total_DATABYTES\\[i\\],1);//发送元件字节数的ASCII代码)datasum_check+total_DATABYTES\\[i\\];}
 
Serial.SendData(&ETX_end,1);//发送“结束”标志代码senddatasum_CHECK+ETX_end;
 
Change_to_ASCII(senddatasum_CHECK,senddatasum_CHECK);//将“和”转化成ASCII码
 
for (i=0;i<2;i++) Serial.SendData(&senddatasum_CHECK\\[i\\],1);
 
Sleep(1000);//等待PLC响应
 
Serial.ReadData(&read_BUFFER,1);
 
if(read_BUFFER==STX_start){
 
readdata_sum=0;
 
for(i=0;i<2*Read_bytes;i++){Serial.ReadData(&Read_char\\[i\\],1);//读Read_bytes个字节readdata_sum+Read_char\\[i\\];}
 
Serial.ReadData(&read_BUFFER,1);
 
if(read_BUFFER==ETX_end){Serial.ReadData(readdatasum_CHECK,2);//读入的“和”的低2位ASCII码Readdata_sum+=ETX_end;}
 
Change_to_ASCII(readdatasum_check,readdata_sum);//将计算得到的“和”转化成ASCII码
 
if(*readdatasum_CHECK==*readdatasum_check)//“和”校验
 
{ AfxMessageBox(“数据读出成功!”)return TRUE;}
 
else { AfxMessageBox(“校验错误”)return FALSE.}
 
}
 
(2)位元件或字元件状态写操作
 
操作对象元件:同3(1);命令格式:
 
说明:①为写命令起始标志STX,代码为OX02;
 
②为位元件或字元件状态写命令CMD1,命令代码为OX31;
 
③为写位元件或字元件的4位起始地址,高位先发,低位后发,且是以ASCII码的形式发送;
 
④为一次写入位元件或字元件的个数,以ASCII码的形式发送;
 
⑤为待写到PLC RAM区的数据DATA,以ASCII码形式发送;
 
⑥为停止位标志ETX,代码为OX03;
 
⑦为2位和校验,和累计为②、③、④项代码,取其和最低两位转化成ASCII码,高位先发,低位后发。
 
VC++语言实现:
 
BOOL CNTJDlg::WritePLC(char *data_ADDRESS,char *Write_ASC,int bytesnumber)
 
{
 
char total_BYTES\\[2\\];char senddatasum_CHECK\\[2\\];
 
char read_BUFFER;char read_finishBUFFER;
 
int datasum_check=0; int i=0;
 
Serial.SendData(&STX_start,1);//向PLC发送“开始”标志代码
 
datasum_check=0;Serial.SendData(&CMD1_write,1);//发送“写”命令代码
 
datasum_check+CMD1_write;
 
for(i=0;i<4;i++) {Serial.SendData(&data_ADDRESS\\[i\\],1);//发送起始元件地址的ASCII码
 
datasum_check+=data_ADDRESS\\[i\\];
 
Change_to_ASCII(total_DATABYTES,bytesnumber);//将字节数转化成ASCII码
 
for(i=0;i<2;i++)
 
{
 
Serial.SendData(&total_BYTES\\,1);//发送元件字节数的ASCII代码
 
datasum_check+=total_BYTES\\[i\\];}
 
for {i=0;i {
 
Serial.SendData(&Write_ASC\\[i\\],1);//发送要写入的数据的ASCII码
 
datasum_check+=Write_ASC\\[i\\];}
 
Serial.SendData(&ETX_end,1);//发送“结束”标志代码
 
datasum_check+=ETX_end;
 
Change_to_ASCII(senddatasum_CHECK,datasum_check);//将“和”转化成ASCII码
 
Serial.SendData(&senddatasum_CHECK,2);
 
Sleep(1000); Serial.ReadData(&read_finishBUFFER,1);
 
if (read_finishBUFFER==ACK_reply)
 
{AfxMessageBox(“数据写入 OK”)return TRUE;}
 
else {AfxMessageBox(“数据写入失败”)return FALSE。}
 
(3)位元件强制ON操作
 
操作对象:X、Y、M、S、T、C元件;
 
命令格式:
 
说明:①为强制OFF命令起始标志STX,代码为OX02;
 
②为强制OFF命令CMD8,命令代码为OX38H;
 
③为强制OFF位元件4位起始地址,高位先发,低位后发,以ASCII码形式发送;
 
④为停止位标志ETX,代码为OX03;
 
⑤为2位和校验,和累计为②、③、④项代码,取其和最低两位转化成ASCII码,高位先发,低位后发。
 
VC++语言实现:
 
void NTJDlg::ForceOffOperation (char *OFF_Address)
 
{
 
int i;
 
char Sum_Check\\[2\\];
 
char read_buffer;
 
int Sum=0;
 
Serial.SendData(&STX_start,1);//向PLC发送“开始”标志代码
 
Serial.SendData(&CMD8_ForceOFF,1);//发送“OFF”命令代码
 
Sum=CMD8_ForceOFF;
 
for (i=0;i<4;i++) {
 
Serial.SendData(&OFF_Address\\[i\\],1);//发送起始元件地址的ASCII码
 
Sum+=OFF_Address\\[i\\];}
 
Serial.SendData(&ETX_end,1);//发送“结束”标志代码
 
Sum+=ETX_end;
 
Change_to_ASCII(Sum_Check,Sum);//将“和”转化成ASCII码
 
Serial.SendData(&Sum_Check,2);
 
Skeeo(1000);
 
Serial.ReadData(&read_buffer,1);
 
if(read_fininhBUFFER==ACK_reply) AfxMessageBox(“OFF 操作 OK ”);
 
else AfxMessageBox(“OFF 操作失败”)。
 

 
注意:必须严格按照上述四种操作命令格式进行发送,在发送前,起始地址、数据、数据个数、校验和都必须按位转换成ASCII码。从PLC读到的数据亦是ASCII码形式,需要经过适当转换才能利用。另外,要注意强制命令地址与读写地址的顺序不是一样,且一次最多只能传送64个字节数据。
 
4 结论
 

 
利用上述四种操作命令,就可对PLC的RAM区数据进行管理操作。将PLC的工作状态纳入微型计算机管理之下。在此基础上,用户可以应用VC很方便地设计自己的PLC人机接口界面,为监控与管理PLC的运行提供一种良好的方法。