C#实现三菱FX-3U SerialOverTcp

设备信息

测试结果

D值测试

 Y值写入后读取测试

 协议解析

三菱FX 3U系列PLC的通信协议

1. 每次给PLC发送指令后,必须等待PLC的应答完成才能发送下一条指令;
2. 报文都是十六进制ASCII码的形式
3. 相关指令

指令 命令码(ASCII码) 操作原件
读 0(30H) X,Y,M,S,T,C,D
写 1(31H) X,Y,M,S,T,C,D
置位 7(37H) X,Y,M,S,T,C
复位 8(38H) X,Y,M,S,T,C

地址换算:D123这个地址写入数据,那么地址为: address = 123*2 + 4096 = 4342 = 10F6
==================================================================================
读指令

上位机请求:STX(1) + CMD(1) + Address(4) + Length(2) + ETX(1) + SUM(2,从cmd到etx)

PLC响应:STX(1) + 值(n字节) + ETX(1) + SUM(2)

**********************************************************************************
例子:
X,Y,D通过相应的地址换算成新的地址

读取Y005 / Y006 bool

02 30 30 30 41 30 30 31 03 36 35 :范围(0-7 地址) 地址: A0 160
02 30 30 30 41 31 30 31 03 36 36 :范围(10-17 地址) 地址: A1 161
02 30 30 30 41 32 30 31 03 36 37 :范围(20-27 地址) 地址: A2 162

02 30 30 30 41 30 30 31 03 36 35

02 32 30 03 36 35:5亮 32 30 -> 20H -> 转二进制 0010 0000

02 36 30 03 36 39:5,6亮 36 30 -> 60H -> 转二进制 0110 0000


D123读取2个字节,short类型
新版:指令45
老版:
02 30 31 30 46 36 30 32 03 37 32
02 30 30 30 30 03 43 33 -- 值为0
02 30 31 30 30 03 43 34 -- 值为1

==================================================================================
写指令

上位机请求:STX(1) + CMD(1) + Address(4) + Length(2) + Data(4*n)+ ETX(1) + SUM(2,从cmd到etx)

PLC响应:STX(1) + 值(1字节 正确:06H;错误:15H) + ETX(1) + SUM(2)

例子:

Y006设置true
02 37 30 36 30 35 03 30 35
06

D123写入2个字节,short类型,值1

新版
0245313034304636303230313030034143
06

老版:写入值:1
02 31 31 30 46 36 30 34 30 31 30 30 30 30 30 30 03 46 36
06

================================================

核心代码

复制代码

using MelsecFxOverTcp;
using System;
using System.Net.Sockets;
using System.Text;namespace MelsecFxSerialOverTcp.Util
{class MelsecFx{string ip = string.Empty;int port = 0;int SendTimeout = 2000;int ReceiveTimeout = 2000;public MelsecFx(string ip, int port){this.ip = ip;this.port = port;}static string NotSupportedDataType => "输入的类型不支持,请重新输入";/// <summary>/// 地址解析/// </summary>static void FxAnalysisAddress(string address, ref MelsecMcDataType Content1, ref ushort Content2){switch (address[0]){case 'M':case 'm':Content1 = MelsecMcDataType.M;Content2 = Convert.ToUInt16(address.Substring(1), MelsecMcDataType.M.FromBase);break;case 'X':case 'x':Content1 = MelsecMcDataType.X;Content2 = Convert.ToUInt16(address.Substring(1), 8);break;case 'Y':case 'y':Content1 = MelsecMcDataType.Y;Content2 = Convert.ToUInt16(address.Substring(1), 8);break;case 'D':case 'd':Content1 = MelsecMcDataType.D;Content2 = Convert.ToUInt16(address.Substring(1), MelsecMcDataType.D.FromBase);break;case 'S':case 's':Content1 = MelsecMcDataType.S;Content2 = Convert.ToUInt16(address.Substring(1), MelsecMcDataType.S.FromBase);break;case 'T':case 't':if (address[1] == 'N' || address[1] == 'n'){Content1 = MelsecMcDataType.TN;Content2 = Convert.ToUInt16(address.Substring(2), MelsecMcDataType.TN.FromBase);break;}if (address[1] == 'S' || address[1] == 's'){Content1 = MelsecMcDataType.TS;Content2 = Convert.ToUInt16(address.Substring(2), MelsecMcDataType.TS.FromBase);break;}if (address[1] == 'C' || address[1] == 'c'){Content1 = MelsecMcDataType.TC;Content2 = Convert.ToUInt16(address.Substring(2), MelsecMcDataType.TC.FromBase);break;}throw new Exception(NotSupportedDataType);case 'C':case 'c':if (address[1] == 'N' || address[1] == 'n'){Content1 = MelsecMcDataType.CN;Content2 = Convert.ToUInt16(address.Substring(2), MelsecMcDataType.CN.FromBase);break;}if (address[1] == 'S' || address[1] == 's'){Content1 = MelsecMcDataType.CS;Content2 = Convert.ToUInt16(address.Substring(2), MelsecMcDataType.CS.FromBase);break;}if (address[1] == 'C' || address[1] == 'c'){Content1 = MelsecMcDataType.CC;Content2 = Convert.ToUInt16(address.Substring(2), MelsecMcDataType.CC.FromBase);break;}throw new Exception(NotSupportedDataType);default:throw new Exception(NotSupportedDataType);}}public bool ConnectServer(){bool ret = false;TcpClient client = null;try{using (client = new TcpClient(ip, port)){ret = client.Connected;client.Close();}}catch (Exception ex){}finally{if (null != client) client.Close();}return ret;}/// <summary>/// // 串口或者网口发送数据/// </summary>/// <exception cref="Exception"></exception>byte[] SendWaitResponse(byte[] data){var requestStr = DataHelper.ToHexString(data, data.Length, true);DataMgr.MainUI.AddMessage("C -> S: " + requestStr);byte[] ret = null;using (var client = new TcpClient(ip, port)){client.SendTimeout = SendTimeout;client.ReceiveTimeout = ReceiveTimeout;var netstream = client.GetStream();//netstream.Write(data, 0, data.Length);//byte[] temp = new byte[2048];int recvnum = netstream.Read(temp, 0, temp.Length);if (recvnum == 0){throw new Exception("数据接收超时");}ret = new byte[recvnum];Array.Copy(temp, 0, ret, 0, recvnum);}var responseStr = DataHelper.ToHexString(ret, ret.Length, true);DataMgr.MainUI.AddMessage("S -> C: " + responseStr);return ret;}public bool[] ReadBool(string address, int length){Console.WriteLine($"ReadBool,address={address},length={length}");MelsecMcDataType Content1 = MelsecMcDataType.M;ushort Content2 = 0;FxAnalysisAddress(address, ref Content1, ref Content2);// 地址转换ushort content = Content2;if (Content1 == MelsecMcDataType.M){content = ((content < 8000) ? ((ushort)((int)content / 8 + 256)) : ((ushort)((content - 8000) / 8 + 480)));}else if (Content1 == MelsecMcDataType.X){content = (ushort)((int)content / 8 + 128);}else if (Content1 == MelsecMcDataType.Y){content = (ushort)((int)content / 8 + 160);}else if (Content1 == MelsecMcDataType.S){content = (ushort)((int)content / 8);}else if (Content1 == MelsecMcDataType.CS){content = (ushort)((int)content / 8 + 448);}else if (Content1 == MelsecMcDataType.CC){content = (ushort)((int)content / 8 + 960);}else if (Content1 == MelsecMcDataType.TS){content = (ushort)((int)content / 8 + 192);}else{if (Content1 != MelsecMcDataType.TC){ throw new Exception("当前的类型不支持位读写");}content = (ushort)((int)content / 8 + 704);}var Content3 = (ushort)((int)Content2 % 8);ushort num = (ushort)((Content2 + length - 1) / 8 - (int)Content2 / 8 + 1);byte[] array = new byte[11]{2,48,SoftBasic.BuildAsciiBytesFrom(content)[0],SoftBasic.BuildAsciiBytesFrom(content)[1],SoftBasic.BuildAsciiBytesFrom(content)[2],SoftBasic.BuildAsciiBytesFrom(content)[3],SoftBasic.BuildAsciiBytesFrom((byte)num)[0],SoftBasic.BuildAsciiBytesFrom((byte)num)[1],3,0,0};DataHelper.FxCalculateSum(array).CopyTo(array, 9); byte[] response = SendWaitResponse(array);// **********************// Y005 or Y006 读取测试//string responseStr = "02 36 30 03 36 39";// 5亮:02 32 30 03 36 35 // 5和6亮:02 36 30 03 36 39//var response = DataHelper.ToHexByte(responseStr); var results = ExtractActualBoolData(response, Content3, length); return results;}public byte[] ReadWord(string address, ushort length, bool isNewVersion = false){Console.WriteLine($"ReadWord,address={address},length={length}");MelsecMcDataType Content1 = MelsecMcDataType.M;ushort Content2 = 0;FxAnalysisAddress(address, ref Content1, ref Content2);ushort content = Content2;if (Content1 == MelsecMcDataType.D){content = ((content < 8000) ? (isNewVersion ? ((ushort)(content * 2 + 16384)) : ((ushort)(content * 2 + 4096))) : ((ushort)((content - 8000) * 2 + 3584)));}else if (Content1 == MelsecMcDataType.CN){content = ((content < 200) ? ((ushort)(content * 2 + 2560)) : ((ushort)((content - 200) * 4 + 3072)));}else{if (Content1 != MelsecMcDataType.TN){ throw new Exception("当前的类型不支持字读写");}content = (ushort)(content * 2 + 2048);}length = (ushort)(length * 2);byte[] array;if (isNewVersion){array = new byte[13]{2,69,48,48,SoftBasic.BuildAsciiBytesFrom(content)[0],SoftBasic.BuildAsciiBytesFrom(content)[1],SoftBasic.BuildAsciiBytesFrom(content)[2],SoftBasic.BuildAsciiBytesFrom(content)[3],SoftBasic.BuildAsciiBytesFrom((byte)length)[0],SoftBasic.BuildAsciiBytesFrom((byte)length)[1],3,0,0};DataHelper.FxCalculateSum(array).CopyTo(array, 11);}else{array = new byte[11]{2,48,SoftBasic.BuildAsciiBytesFrom(content)[0],SoftBasic.BuildAsciiBytesFrom(content)[1],SoftBasic.BuildAsciiBytesFrom(content)[2],SoftBasic.BuildAsciiBytesFrom(content)[3],SoftBasic.BuildAsciiBytesFrom((byte)length)[0],SoftBasic.BuildAsciiBytesFrom((byte)length)[1],3,0,0};DataHelper.FxCalculateSum(array).CopyTo(array, 9);}//var request = DataHelper.ToHexString(array, array.Length, true);//DataMgr.MainUI.AddMessage("request:" + request);// 串口或者网口发送数据// .....// // **********************// D123 读取测试//string responseStr = "02 30 31 30 30 03 43 34";// 值为0:02 30 30 30 30 03 43 33 // 值为1:02 30 31 30 30 03 43 34//var response = DataHelper.ToHexByte(responseStr);//DataMgr.MainUI.AddMessage(responseStr);var response = SendWaitResponse(array);var results = ExtractActualData(response);return results;}public void WriteBool(string address, bool value){Console.WriteLine($"WriteBool,address={address},value={value}");MelsecMcDataType Content1 = MelsecMcDataType.M;ushort Content2 = 0;FxAnalysisAddress(address, ref Content1, ref Content2);  ushort content = Content2;if (Content1 == MelsecMcDataType.M){content = ((content < 8000) ? ((ushort)(content + 2048)) : ((ushort)(content - 8000 + 3840)));}else if (Content1 == MelsecMcDataType.S){content = content;}else if (Content1 == MelsecMcDataType.X){content = (ushort)(content + 1024);}else if (Content1 == MelsecMcDataType.Y){content = (ushort)(content + 1280);}else if (Content1 == MelsecMcDataType.CS){content = (ushort)(content + 448);}else if (Content1 == MelsecMcDataType.CC){content = (ushort)(content + 960);}else if (Content1 == MelsecMcDataType.CN){content = (ushort)(content + 3584);}else if (Content1 == MelsecMcDataType.TS){content = (ushort)(content + 192);}else if (Content1 == MelsecMcDataType.TC){content = (ushort)(content + 704);}else{if (Content1 != MelsecMcDataType.TN){// "当前的类型不支持位读写"return ;}content = (ushort)(content + 1536);}byte[] array = new byte[9]{2,(byte)(value ? 55 : 56),SoftBasic.BuildAsciiBytesFrom(content)[2],SoftBasic.BuildAsciiBytesFrom(content)[3],SoftBasic.BuildAsciiBytesFrom(content)[0],SoftBasic.BuildAsciiBytesFrom(content)[1],3,0,0};DataHelper.FxCalculateSum(array).CopyTo(array, 7);SendWaitResponse(array);}//public static void Write(string address, int value)//{//    Write(address, new int[1] { value });//}//public static void Write(string address, int[] values)//{//    Write(address, ByteTransformBase.TransByte(values));//}public void WriteWord(string address, byte[] value, bool isNewVersion = false){Console.WriteLine($"WriteBytes,address={address},value={value}");MelsecMcDataType Content1 = MelsecMcDataType.M;ushort Content2 = 0;FxAnalysisAddress(address, ref Content1, ref Content2);ushort content = Content2;if (Content1 == MelsecMcDataType.D){content = ((content < 8000) ? (isNewVersion ? ((ushort)(content * 2 + 16384)) : ((ushort)(content * 2 + 4096))) : ((ushort)((content - 8000) * 2 + 3584)));}else if (Content1 == MelsecMcDataType.CN){content = ((content < 200) ? ((ushort)(content * 2 + 2560)) : ((ushort)((content - 200) * 4 + 3072)));}else{if (Content1 != MelsecMcDataType.TN){return;// 当前的类型不支持字读写}content = (ushort)(content * 2 + 2048);}if (value != null){value = SoftBasic.BuildAsciiBytesFrom(value);}byte[] array = null;if (isNewVersion){array = new byte[13 + value.Length];array[0] = 2;array[1] = 69;array[2] = 49;array[3] = 48;array[4] = SoftBasic.BuildAsciiBytesFrom(content)[0];array[5] = SoftBasic.BuildAsciiBytesFrom(content)[1];array[6] = SoftBasic.BuildAsciiBytesFrom(content)[2];array[7] = SoftBasic.BuildAsciiBytesFrom(content)[3];array[8] = SoftBasic.BuildAsciiBytesFrom((byte)(value.Length / 2))[0];array[9] = SoftBasic.BuildAsciiBytesFrom((byte)(value.Length / 2))[1];Array.Copy(value, 0, array, 10, value.Length);array[array.Length - 3] = 3; }else{array = new byte[11 + value.Length];array[0] = 2;array[1] = 49;array[2] = SoftBasic.BuildAsciiBytesFrom(content)[0];array[3] = SoftBasic.BuildAsciiBytesFrom(content)[1];array[4] = SoftBasic.BuildAsciiBytesFrom(content)[2];array[5] = SoftBasic.BuildAsciiBytesFrom(content)[3];array[6] = SoftBasic.BuildAsciiBytesFrom((byte)(value.Length / 2))[0];array[7] = SoftBasic.BuildAsciiBytesFrom((byte)(value.Length / 2))[1];Array.Copy(value, 0, array, 8, value.Length);array[array.Length - 3] = 3;}DataHelper.FxCalculateSum(array).CopyTo(array, array.Length - 2);SendWaitResponse(array); }public static string CheckPlcReadResponse(byte[] ack){if (ack.Length == 0){return "接收的数据长度为0";}if (ack[0] == 21){return "PLC反馈的数据无效,Actual: " + SoftBasic.ByteToHexString(ack, ' ');}if (ack[0] != 2){return "PLC反馈信号错误:" + ack[0] + " Actual: " + SoftBasic.ByteToHexString(ack, ' ');}if (!DataHelper.CheckSum(ack)){return "PLC反馈报文的和校验失败!";}return string.Empty;}public static byte[] ExtractActualData(byte[] response){byte[] array = new byte[(response.Length - 4) / 2];for (int i = 0; i < array.Length; i++){byte[] bytes = new byte[2]{response[i * 2 + 1],response[i * 2 + 2]};array[i] = Convert.ToByte(Encoding.ASCII.GetString(bytes), 16);}return array;}public static bool[] ExtractActualBoolData(byte[] response, int start, int length){// 02 32 30 03 36 35 Data:20H -> 十进制32 -> 0010 0000// 02 36 30 03 36 39 Data:60H -> 十进制96 -> 0110 0000byte[] Content = ExtractActualData(response);bool[] arraybool = new bool[length];bool[] array2 = SoftBasic.ByteToBoolArray(Content, Content.Length * 8);// false false false false true false falsefor (int i = 0; i < length; i++){arraybool[i] = array2[i + start];}return arraybool;}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/60745.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

剑指offer39.数组中出现次数超过一半的数字

这个题非常简单&#xff0c;解法有很多种&#xff0c;我用的是HashMap记录每个元素出现的次数&#xff0c;只要次数大于数组长度的一半就返回。下面是我的代码&#xff1a; class Solution {public int majorityElement(int[] nums) {int len nums.length/2;HashMap<Integ…

【Github】Uptime Kuma:自托管监控工具的完美选择

简介&#xff1a; Uptime Kuma 是一款强大的自托管监控工具&#xff0c;通过简单的部署和配置&#xff0c;可以帮助你监控服务器、VPS 和其他网络服务的在线状态。相比于其他类似工具&#xff0c;Uptime Kuma 提供更多的灵活性和自由度。本文将介绍 Uptime Kuma 的功能、如何使…

图解java.util.concurrent并发包源码系列——深入理解AQS,看完可以吊打面试官

图解java.util.concurrent并发包源码系列——深入理解AQS&#xff0c;看完可以吊打面试官 AQS是什么&#xff1f;有什么作用&#xff1f;AQS的原理自定义资源资源的获取与释放线程阻塞等待唤醒 AQS源码核心成员变量Node 的内部结构waitStatusprev、next、threadnextWaiterprede…

第十四届中国大学生服务外包大赛圆满落幕,合合信息助力人才发展消除市场“信息差”

老年人存在记账难题&#xff0c;如何通过技术手段处理&#xff1f;已经上线多年的软件产品&#xff0c;如何优化才能更符合现代人群的“胃口”&#xff1f;这些微小却关键的问题颇具社会价值&#xff0c;青年学子们的参与或许能够打开新的产品构建维度。 近日&#xff0c;“中…

从前序与中序遍历序列构造二叉树,从中序与后序遍历序列构造二叉树

目录 从前序与中序遍历序列构造二叉树从中序与后序遍历序列构造二叉树 从前序与中序遍历序列构造二叉树 题目链接 给定两个整数数组 preorder 和 inorder &#xff0c;其中 preorder 是二叉树的先序遍历&#xff0c; inorder 是同一棵树的中序遍历&#xff0c;请构造二叉树并返…

【算法|数组】滑动窗口

算法|数组——滑动窗口 引入 给定一个含有 n 个正整数的数组和一个正整数 target 。 找出该数组中满足其和 ≥ target 的长度最小的 连续子数组 [numsl, numsl1, ..., numsr-1, numsr] &#xff0c;并返回其长度**。**如果不存在符合条件的子数组&#xff0c;返回 0 。 示例…

【Node.js】低代码平台源码

一、低代码简介 低代码管理系统是一种通过可视化界面和简化的开发工具&#xff0c;使非专业开发人员能够快速构建和管理应用程序的系统。它提供了一套预先定义的组件和模块&#xff0c;使用户可以通过拖放操作来设计应用程序的界面和逻辑。低代码管理系统还提供了自动化的工作…

2. Linux安装Git

yum安装 查看版本 版本太低&#xff0c;所以我们采用自己上传编译的方式进行 删除已安装的git yum remove git 下载最新安装包&#xff0c;并上传到服务器文件夹下 上传&#xff0c;解压 5.安装编译需要的依赖 yum install curl-devel expat-devel gettext-devel openssl-…

【面试八股文】每日一题:谈谈你对异常的理解

每日一题-Java核心-谈谈你对异常的理解【面试八股文】 异常是程序在运行过程中出现的错误或不正常的情况。当程序执行过程中遇到无法处理的错误或者不符合预期的情况&#xff0c;就会抛出异常。异常可以分为两种类型&#xff1a;受检异常和非受检异常。 受检异常是指在程序编译…

Element组件浅尝辄止2:Card卡片组件

根据官方说法&#xff1a; 将信息聚合在卡片容器中展示。 1.啥时候使用&#xff1f;When? 既然是信息聚合的容器&#xff0c;那场景就好说了 新建页面时可以用来当做页面容器页面的某一部分&#xff0c;可以用来当做子容器 2.怎样使用&#xff1f;How&#xff1f; //Card …

Django实现音乐网站 ⑺

使用Python Django框架制作一个音乐网站&#xff0c; 本篇主要是后台对歌手原有实现功能的基础上进行优化处理。 目录 新增编辑 表字段名称修改 隐藏单曲、专辑数 姓名首字母 安装xpinyin 获取姓名首字母 重写保存方法 列表显示 图片显示处理 引入函数 路径改为显示…

rust关于项目结构包,Crate和mod和目录的组织

rust 最近开始学习rust语言。感觉这门语言相对java确实是难上很多。开几个文章把遇到的问题记录一下 rust关于包&#xff0c;Crate 关于包&#xff0c;Crate这块先看看官方书籍怎么说的 crate 是 Rust 在编译时最小的代码单位。如果你用 rustc 而不是 cargo 来编译一个文件…