真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

ModbusRTU通信工具設(shè)計(jì)

 

Modbus 是一個(gè)工業(yè)上常用的通訊協(xié)議、一種通訊約定。

成都創(chuàng)新互聯(lián)是一家集網(wǎng)站建設(shè),威海企業(yè)網(wǎng)站建設(shè),威海品牌網(wǎng)站建設(shè),網(wǎng)站定制,威海網(wǎng)站建設(shè)報(bào)價(jià),網(wǎng)絡(luò)營(yíng)銷,網(wǎng)絡(luò)優(yōu)化,威海網(wǎng)站推廣為一體的創(chuàng)新建站企業(yè),幫助傳統(tǒng)企業(yè)提升企業(yè)形象加強(qiáng)企業(yè)競(jìng)爭(zhēng)力??沙浞譂M足這一群體相比中小企業(yè)更為豐富、高端、多元的互聯(lián)網(wǎng)需求。同時(shí)我們時(shí)刻保持專業(yè)、時(shí)尚、前沿,時(shí)刻以成就客戶成長(zhǎng)自我,堅(jiān)持不斷學(xué)習(xí)、思考、沉淀、凈化自己,讓我們?yōu)楦嗟钠髽I(yè)打造出實(shí)用型網(wǎng)站。

ModBus 協(xié)議是應(yīng)用層報(bào)文傳輸協(xié)議(OSI 模型第7層),它定義了一個(gè)與通信層無(wú)關(guān)的協(xié)議數(shù)據(jù)單元(PDU),即PDU=功能碼+數(shù)據(jù)域。
ModBus 協(xié)議能夠應(yīng)用在不同類型的總線或網(wǎng)絡(luò)。對(duì)應(yīng)不同的總線或網(wǎng)絡(luò),Modbus 協(xié)議引入一些附加域映射成應(yīng)用數(shù)據(jù)單元(ADU),即ADU=附加域+PDU。目前,Modbus 有下列三種通信方式:
1. 以太網(wǎng),對(duì)應(yīng)的通信模式是Modbus TCP。
2. 異步串行傳輸(各種介質(zhì)如有線RS-232-/422/485/;光纖、無(wú)線等),對(duì)應(yīng)的通信模式是 Modbus RTU 或 Modbus ASCII。Modbus 的ASCII、RTU 協(xié)議規(guī)定了消息、數(shù)據(jù)的結(jié)構(gòu)、命令和應(yīng)答的方式,數(shù)據(jù)通訊采用Maser/Slave方式。
3. 高速令牌傳遞網(wǎng)絡(luò),對(duì)應(yīng)的通信模式是Modbus PLUS。

 

Modbus 需要對(duì)數(shù)據(jù)進(jìn)行校驗(yàn),串行協(xié)議中除有奇偶校驗(yàn)外,ASCII 模式采用LRC 校驗(yàn);RTU 模式采用16位CRC 校驗(yàn);TCP 模式?jīng)]有額外規(guī)定校驗(yàn),因?yàn)門CP 是一個(gè)面向連接的可靠協(xié)議。

 

Modbus 協(xié)議的應(yīng)用中,最常用的是Modbus RTU 傳輸模式。

 

RTU 傳輸模式 

當(dāng)設(shè)備使用RTU (Remote Terminal Unit) 模式在 Modbus  串行鏈路通信, 報(bào)文中每個(gè)8位字節(jié)含有兩個(gè)4位十六進(jìn)制字符。這種模式的主要優(yōu)點(diǎn)是較高的數(shù)據(jù)密度,在相同的波特率下比ASCII 模式有更高的吞吐率。每個(gè)報(bào)文必須以連續(xù)的字符流傳送。 

 

RTU 模式每個(gè)字節(jié) ( 11 位 ) 的格式為:

       編碼系統(tǒng):  8位二進(jìn)制。 報(bào)文中每個(gè)8位的字節(jié)含有兩個(gè)4位十六進(jìn)制字符(0–9, A–F)

  Bits per Byte:  1 起始位

                   8 數(shù)據(jù)位, 首先發(fā)送最低有效位

                   1 位作為奇偶校驗(yàn)

                   1 停止位

偶校驗(yàn)是要求的,其它模式 ( 奇校驗(yàn), 無(wú)校驗(yàn) ) 也可以使用。為了保證與其它產(chǎn)品的最大兼容性,同時(shí)支持無(wú)校驗(yàn)?zāi)J绞墙ㄗh的。默認(rèn)校驗(yàn)?zāi)J侥J?nbsp;必須為偶校驗(yàn)。注:使用無(wú)校驗(yàn)要求2 個(gè)停止位。 

 

字符的串行傳送方式:

每個(gè)字符或字節(jié)均由此順序發(fā)送(從左到右):最低有效位 (LSB) . . . 最高有效位 (MSB)

Modbus RTU 通信工具設(shè)計(jì)

圖1:RTU 模式位序列 

 

設(shè)備配置為奇校驗(yàn)、偶校驗(yàn)或無(wú)校驗(yàn)都可以接受。如果無(wú)奇偶校驗(yàn),將傳送一個(gè)附加的停止位以填充字符幀:

Modbus RTU 通信工具設(shè)計(jì)

圖2:RTU 模式位序列 (無(wú)校驗(yàn)的特殊情況)

 

幀檢驗(yàn)域:循環(huán)冗余校驗(yàn) (CRC)

在RTU 模式包含一個(gè)對(duì)全部報(bào)文內(nèi)容執(zhí)行的,基于循環(huán)冗余校驗(yàn) (CRC - Cyclical Redundancy Checking) 算法的錯(cuò)誤檢驗(yàn)域。

CRC 域檢驗(yàn)整個(gè)報(bào)文的內(nèi)容。不管報(bào)文有無(wú)奇偶校驗(yàn),均執(zhí)行此檢驗(yàn)。

CRC 包含由兩個(gè)8位字節(jié)組成的一個(gè)16位值。  

CRC 域作為報(bào)文的最后的域附加在報(bào)文之后。計(jì)算后,首先附加低字節(jié),然后是高字節(jié)。CRC 高字節(jié)為報(bào)文發(fā)送的最后一個(gè)子節(jié)。

附加在報(bào)文后面的CRC 的值由發(fā)送設(shè)備計(jì)算。接收設(shè)備在接收?qǐng)?bào)文時(shí)重新計(jì)算 CRC 的值,并將計(jì)算結(jié)果于實(shí)際接收到的CRC 值相比較。如果兩個(gè)值不相等,則為錯(cuò)誤。

CRC 的計(jì)算,開始對(duì)一個(gè)16位寄存器預(yù)裝全1。 然后將報(bào)文中的連續(xù)的8位子節(jié)對(duì)其進(jìn)行后續(xù)的計(jì)算。只有字符中的8個(gè)數(shù)據(jù)位參與生成CRC 的運(yùn)算,起始位,停止位和校驗(yàn)位不參與 CRC 計(jì)算。

CRC 的生成過(guò)程中, 每個(gè) 8–位字符與寄存器中的值異或。然后結(jié)果向最低有效位(LSB)方向移動(dòng)(Shift) 1位,而最高有效位(MSB)位置充零。 然后提取并檢查 LSB:如果LSB 為1, 則寄存器中的值與一個(gè)固定的預(yù)置值異或;如果LSB 為 0, 則不進(jìn)行異或操作。

這個(gè)過(guò)程將重復(fù)直到執(zhí)行完8次移位。完成最后一次(第8次)移位及相關(guān)操作后,下一個(gè)8位字節(jié)與寄存器的當(dāng)前值異或,然后又同上面描述過(guò)的一樣重復(fù)8次。當(dāng)所有報(bào)文中子節(jié)都運(yùn)算之后得到的寄存器中的最終值,就是CRC。

當(dāng)CRC 附加在報(bào)文之后時(shí),首先附加低字節(jié),然后是高字節(jié)。

CRC 算法如下:

  1. private bool CheckResponse(byte[] response) 
  2.     //Perform a basic CRC check: 
  3.     byte[] CRC = new byte[2]; 
  4.     GetCRC(response, ref CRC); 
  5.     if (CRC[0] == response[response.Length - 2] && CRC[1] == response[response.Length - 1]) 
  6.     return true; 
  7.     else 
  8.     return false; 
  9.  
  10. private void GetCRC(byte[] message, ref byte[] CRC) 
  11.     //Function expects a modbus message of any length as well as a 2 byte CRC array in which to  
  12.     //return the CRC values: 
  13.  
  14.     ushort CRCFull = 0xFFFF; 
  15.     byte CRCHigh = 0xFF, CRCLow = 0xFF; 
  16.     char CRCLSB; 
  17.  
  18.     for (int i = 0; i < (message.Length) - 2; i++) 
  19.     { 
  20.     CRCFull = (ushort)(CRCFull ^ message[i]); 
  21.  
  22.     for (int j = 0; j < 8; j++) 
  23.     { 
  24.         CRCLSB = (char)(CRCFull & 0x0001); 
  25.         CRCFull = (ushort)((CRCFull >> 1) & 0x7FFF); 
  26.  
  27.         if (CRCLSB == 1) 
  28.         CRCFull = (ushort)(CRCFull ^ 0xA001); 
  29.     } 
  30.     } 
  31.     CRC[1] = CRCHigh = (byte)((CRCFull >> 8) & 0xFF); 
  32.     CRC[0] = CRCLow = (byte)(CRCFull & 0xFF); 

 

幀描述 (如下圖所示) :

Modbus RTU 通信工具設(shè)計(jì)

圖3:RTU 報(bào)文幀

注意:Modbus  RTU 幀最大為256字節(jié)。

 

下面是我為公司設(shè)計(jì)的一個(gè) Modbus RTU 通信測(cè)試小工具,界面截圖如下:

Modbus RTU 通信工具設(shè)計(jì)

圖4:Modbus RTU 通信工具

 

我的通用Modbus RTU 動(dòng)態(tài)庫(kù),modbus.cs 如下:

  1. modbus.cs 
  2.  
  3. using System; 
  4. using System.Collections.Generic; 
  5. using System.Text; 
  6. using System.IO.Ports; 
  7. using System.Threading; 
  8.  
  9. namespace SerialPort_Lib 
  10.     public class modbus 
  11.     { 
  12.         private SerialPort sp = new SerialPort(); 
  13.         public string modbusStatus; 
  14.  
  15.         #region Constructor / Deconstructor 
  16.         public modbus() 
  17.         { 
  18.         } 
  19.         ~modbus() 
  20.         { 
  21.         } 
  22.         #endregion 
  23.  
  24.         #region Open / Close Procedures 
  25.         public bool Open(string portName, int baudRate, int databits, Parity parity, StopBits stopBits) 
  26.         { 
  27.             //Ensure port isn't already opened: 
  28.             if (!sp.IsOpen) 
  29.             { 
  30.                 //Assign desired settings to the serial port: 
  31.                 sp.PortName = portName; 
  32.                 sp.BaudRate = baudRate; 
  33.                 sp.DataBits = databits; 
  34.                 sp.Parity = parity; 
  35.                 sp.StopBits = stopBits; 
  36.                 //These timeouts are default and cannot be editted through the class at this point: 
  37.                 sp.ReadTimeout = -1; 
  38.                 sp.WriteTimeout = 10000; 
  39.  
  40.                 try 
  41.                 { 
  42.                     sp.Open(); 
  43.                 } 
  44.                 catch (Exception err) 
  45.                 { 
  46.                     modbusStatus = "Error opening " + portName + ": " + err.Message; 
  47.                     return false; 
  48.                 } 
  49.                 modbusStatus = portName + " opened successfully"; 
  50.                 return true; 
  51.             } 
  52.             else 
  53.             { 
  54.                 modbusStatus = portName + " already opened"; 
  55.                 return false; 
  56.             } 
  57.         } 
  58.         public bool Close() 
  59.         { 
  60.             //Ensure port is opened before attempting to close: 
  61.             if (sp.IsOpen) 
  62.             { 
  63.                 try 
  64.                 { 
  65.                     sp.Close(); 
  66.                 } 
  67.                 catch (Exception err) 
  68.                 { 
  69.                     modbusStatus = "Error closing " + sp.PortName + ": " + err.Message; 
  70.                     return false; 
  71.                 } 
  72.                 modbusStatus = sp.PortName + " closed successfully"; 
  73.                 return true; 
  74.             } 
  75.             else 
  76.             { 
  77.                 modbusStatus = sp.PortName + " is not open"; 
  78.                 return false; 
  79.             } 
  80.         } 
  81.         #endregion 
  82.  
  83.         #region CRC Computation 
  84.         private void GetCRC(byte[] message, ref byte[] CRC) 
  85.         { 
  86.             //Function expects a modbus message of any length as well as a 2 byte CRC array in which to  
  87.             //return the CRC values: 
  88.  
  89.             ushort CRCFull = 0xFFFF; 
  90.             byte CRCHigh = 0xFF, CRCLow = 0xFF; 
  91.             char CRCLSB; 
  92.  
  93.             for (int i = 0; i < (message.Length) - 2; i++) 
  94.             { 
  95.                 CRCFull = (ushort)(CRCFull ^ message[i]); 
  96.  
  97.                 for (int j = 0; j < 8; j++) 
  98.                 { 
  99.                     CRCLSB = (char)(CRCFull & 0x0001); 
  100.                     CRCFull = (ushort)((CRCFull >> 1) & 0x7FFF); 
  101.  
  102.                     if (CRCLSB == 1) 
  103.                         CRCFull = (ushort)(CRCFull ^ 0xA001); 
  104.                 } 
  105.             } 
  106.             CRC[1] = CRCHigh = (byte)((CRCFull >> 8) & 0xFF); 
  107.             CRC[0] = CRCLow = (byte)(CRCFull & 0xFF); 
  108.         } 
  109.         #endregion 
  110.  
  111.         #region Build Message 
  112.         private void BuildMessage(byte address, byte type, ushort start, ushort registers, ref byte[] message) 
  113.         { 
  114.             //Array to receive CRC bytes: 
  115.             byte[] CRC = new byte[2]; 
  116.  
  117.             message[0] = address; 
  118.             message[1] = type; 
  119.             message[2] = (byte)(start >> 8); 
  120.             message[3] = (byte)start; 
  121.             message[4] = (byte)(registers >> 8); 
  122.             message[5] = (byte)registers; 
  123.  
  124.             GetCRC(message, ref CRC); 
  125.             message[message.Length - 2] = CRC[0]; 
  126.             message[message.Length - 1] = CRC[1]; 
  127.         } 
  128.         #endregion 
  129.  
  130.         #region Check Response 
  131.         private bool CheckResponse(byte[] response) 
  132.         { 
  133.             //Perform a basic CRC check: 
  134.             byte[] CRC = new byte[2]; 
  135.             GetCRC(response, ref CRC); 
  136.             if (CRC[0] == response[response.Length - 2] && CRC[1] == response[response.Length - 1]) 
  137.                 return true; 
  138.             else 
  139.                 return false; 
  140.         } 
  141.         #endregion 
  142.  
  143.         #region Get Response 
  144.         private void GetResponse(ref byte[] response) 
  145.         { 
  146.             //There is a bug in .Net 2.0 DataReceived Event that prevents people from using this 
  147.             //event as an interrupt to handle data (it doesn't fire all of the time).  Therefore 
  148.             //we have to use the ReadByte command for a fixed length as it's been shown to be reliable. 
  149.             for (int i = 0; i < response.Length; i++) 
  150.             { 
  151.                 response[i] = (byte)(sp.ReadByte()); 
  152.             } 
  153.         } 
  154.         #endregion 
  155.  
  156.         #region GetModbusData 獲得接收數(shù)據(jù) 
  157.         public bool GetModbusData(ref byte[] values) 
  158.         { 
  159.             //Ensure port is open: 
  160.             if (sp.IsOpen) 
  161.             { 
  162.                 // 等待線程進(jìn)入  
  163.                 //Monitor.Enter(sp); 
  164.  
  165.                 //Clear in/out buffers: 
  166.                 //sp.DiscardOutBuffer(); 
  167.                 //sp.DiscardInBuffer(); 
  168.  
  169.                 //Message is 1 addr + 1 type + N Data + 2 CRC 
  170.                      
  171.                 try 
  172.                 { 
  173.                     //GetResponse(ref readBuffer); 
  174.                     //string str = readBuffer.ToString(); 
  175.  
  176.                     int count = sp.BytesToRead; 
  177.                     if (count > 0) 
  178.                     { 
  179.                         byte[] readBuffer = new byte[count]; 
  180.  
  181.                         GetResponse(ref readBuffer); 
  182.  
  183.                         //   readData = new byte[29]; 
  184.                         //   Array.Copy(readBuffer, readData, readData.Length); 
  185.  
  186.                         // CRC 驗(yàn)證 
  187.                         if (CheckResponse(readBuffer)) 
  188.                         { 
  189.                             //顯示輸入數(shù)據(jù) 
  190.                             values = readBuffer; 
  191.  
  192.                             modbusStatus = "Write successful"; 
  193.  
  194.                             sp.DiscardInBuffer(); 
  195.  
  196.                             //values = System.Text.Encoding.ASCII.GetString(readData); 
  197.                             return true; 
  198.                         } 
  199.                         else 
  200.                         { 
  201.                             modbusStatus = "CRC error"; 
  202.  
  203.                             sp.DiscardInBuffer(); 
  204.  
  205.                             return false; 
  206.                         } 
  207.                     } 
  208.                     else return false; 
  209.                 } 
  210.                 catch (Exception err) 
  211.                 { 
  212.                     modbusStatus = "Error in write event: " + err.Message; 
  213.  
  214.                     sp.DiscardInBuffer(); 
  215.  
  216.                     return false; 
  217.                 } 
  218.  
  219.                 //finally 
  220.                 //{ 
  221.                     // 通知其它對(duì)象 
  222.                     //Monitor.Pulse(sp); 
  223.                     // 釋放對(duì)象鎖  
  224.                     //Monitor.Exit(sp); 
  225.                 //} 
  226.             } 
  227.             else 
  228.             { 
  229.                 modbusStatus = "Serial port not open"; 
  230.                 return false; 
  231.             } 
  232.         } 
  233.         #endregion 
  234.  
  235.         #region SendModbusData 打包發(fā)送數(shù)據(jù) 
  236.         public bool SendModbusData(ref byte[] values) 
  237.         { 
  238.             //Ensure port is open: 
  239.             if (sp.IsOpen) 
  240.             { 
  241.                 //Clear in/out buffers: 
  242.                 sp.DiscardOutBuffer(); 
  243.                 sp.DiscardInBuffer(); 
  244.  
  245.                 //Function 3 response buffer: 
  246.                 byte[] response = new byte[values.Length + 2]; 
  247.                 Array.Copy(values, response, values.Length); 
  248.  
  249.                 //BuildMessage(address, (byte)3, start, registers, ref message); 
  250.  
  251.                 //打包帶有 CRC 驗(yàn)證的modbus 數(shù)據(jù)包: 
  252.                 byte[] CRC = new byte[2]; 
  253.                 GetCRC(response, ref CRC); 
  254.                 response[response.Length - 2] = CRC[0]; 
  255.                 response[response.Length - 1] = CRC[1]; 
  256.  
  257.                 values = response; //返回帶有 CRC 驗(yàn)證的modbus 數(shù)據(jù)包 
  258.  
  259.                 //Send modbus message to Serial Port: 
  260.                 try 
  261.                 { 
  262.                     sp.Write(response, 0, response.Length); 
  263.                     //GetResponse(ref response); 
  264.                     return true; 
  265.                 } 
  266.                 catch (Exception err) 
  267.                 { 
  268.                     modbusStatus = "Error in read event: " + err.Message; 
  269.                     return false; 
  270.                 } 
  271.                 //Evaluate message: 
  272.                 //if (CheckResponse(response)) 
  273.                 //{ 
  274.                 //    //Return requested register values: 
  275.                 //    for (int i = 0; i < (response.Length - 5) / 2; i++) 
  276.                 //    { 
  277.                 //        values[i] = response[2 * i + 3]; 
  278.                 //        values[i] <<= 8; 
  279.                 //        values[i] += response[2 * i + 4]; 
  280.                 //    } 
  281.                 //    modbusStatus = "Read successful"; 
  282.                 //    return true; 
  283.                 //} 
  284.                 //else 
  285.                 //{ 
  286.                 //    modbusStatus = "CRC error"; 
  287.                 //    return false; 
  288.                 //} 
  289.             } 
  290.             else 
  291.             { 
  292.                 modbusStatus = "Serial port not open"; 
  293.                 return false; 
  294.             } 
  295.  
  296.         } 
  297.         #endregion 
  298.  
  299.     } 

 

調(diào)用的主要代碼如下:

  1. modbus類的winform調(diào)用代碼 
  2.  
  3. public partial class FormConfig : Form,IModbusData 
  4.     //業(yè)務(wù)處理類 
  5.     B_ModbusData ModbusDataBLL = new B_ModbusData(); 
  6.  
  7.     modbus mb = new modbus(); 
  8.     //SerialPort sp = new SerialPort(); 
  9.     System.Timers.Timer timer = new System.Timers.Timer(); 
  10.  
  11.     public FormConfig() 
  12.     { 
  13.         InitializeComponent(); 
  14.         
  15.         timer.Elapsed += new ElapsedEventHandler(timer_Elapsed); 
  16.     } 
  17.  
  18.     #region Timer Elapsed 事件處理程序 
  19.     bool&
    文章題目:ModbusRTU通信工具設(shè)計(jì)
    網(wǎng)站URL:http://weahome.cn/article/gjpjjp.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部