C#上位机与三菱PLC的通信09---开发自己的通讯库(A-3E版)

1、A-3E报文回顾

 

 具体细节请看:

C#上位机与三菱PLC的通信05--MC协议之QnA-3E报文解析

C#上位机与三菱PLC的通信06--MC协议之QnA-3E报文测试

2、为何要开发自己的通讯库
  

 前面开发了自己的A-1E协议的通讯库,实现了数据的读写,对于封装的通讯库,其实是一个dll文件,请看上节的dll文件,有了这个文件,就可以在项目中直接引用 。

我们只要引用并调用相关的方法即可实现目的, 但写一个通讯库需要非凡的技术,需要考虑的东西很多,比如扩展性,通用性,等等之类的。通过封装通讯库达到更高的层次, 大师就是这样锻造出来的,接下来马上安排A-3E协议的封装,代码是基于上节的基础上添加。

 3、说干就干

1、添加类文件

2、编写核心的通信类A3E.cs

A3E.cs完整代码

using Mitsubishi.Communication.MC.Mitsubishi.Base;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net.Sockets;
using System.Text;
using System.Threading.Tasks;namespace Mitsubishi.Communication.MC.Mitsubishi
{/// <summary>/// A3E报文通讯库/// </summary>public class A3E : MelsecBase{byte _netCode = 0x00, _stationCode = 0x00;public A3E(string ip, short port, byte net_code = 0x00, byte station_code = 0x00) : base(ip, port){_netCode = net_code;_stationCode = station_code;}#region 读数据/// <summary>/// 读取数据/// </summary>/// <typeparam name="T">读取的数据类型</typeparam>/// <param name="address">存储区地址</param>/// <param name="count">读取长度</param>/// <returns></returns>public Result<T> Read<T>(string address, short count){AreaCode areaCode; string start;(areaCode, start) = this.AnalysisAddress(address);return Read<T>(areaCode, start, count);}/// <summary>/// 读取数据/// </summary>/// <typeparam name="T">读取的数据类型</typeparam>/// <param name="areaCode">存储区代码</param>/// <param name="startAddr">开始地址</param>/// <param name="count">读取长度</param>/// <returns></returns>public Result<T> Read<T>(AreaCode areaCode, string startAddr, short count){Result<T> result = new Result<T>();try{// 连接var connectState = this.Connect();if (!connectState.IsSuccessed){throw new Exception(connectState.Message);}// 子命令(位/字)byte readCode = (byte)(typeof(T) == typeof(bool) ? 0x01 : 0x00);//开始地址List<byte> startBytes = this.StartToBytes(areaCode, startAddr);// 读取长度int typeLen = this.CalculatLength<T>();// 读取报文List<byte> bytes = new List<byte>{0x50,0x00,//请求副头部,固定50 00_netCode,// 网络号,根据PLC的设置0xFF,//PLC编号,固定值0xFF,0x03,//目标模块IO编号,固定FF 03_stationCode,// 目标模块站号 0x0C,0x00,  // 剩余字节长度0x0A,0x00,//PLC响应超时时间,以250ms为单位计算 0x01,0x04,// 成批读取readCode,0x00,// 字操作  0x0001 startBytes[0],startBytes[1],startBytes[2],// 起始地址(byte)areaCode,// 区域代码 (byte)(typeLen*count%256),(byte)(typeLen*count/256%256) //长度};//发送报文List<byte> dataBytes = this.Send(bytes, 0);//数据解析result.Datas = this.AnalysisDatas<T>(dataBytes, typeLen);}catch (Exception ex){result = new Result<T>(false, ex.Message);}return result;}#endregion#region 写数据/// <summary>/// 写入数据/// </summary>/// <typeparam name="T">写入的数据类型</typeparam>/// <param name="values">写入的数据列表</param>/// <param name="address">开始地址</param>/// <returns></returns>public Result Write<T>(List<T> values, string address){AreaCode areaCode; string start;(areaCode, start) = this.AnalysisAddress(address);return this.Write<T>(values, areaCode, start);}/// <summary>/// 写入数据/// </summary>/// <typeparam name="T">写入的数据类型</typeparam>/// <param name="values">写入的数据列表</param>/// <param name="areaCode">存储区代码</param>/// <param name="address">开始地址</param>/// <returns></returns>public Result Write<T>(List<T> values, AreaCode areaCode, string startAddr){Result result = new Result();try{// 连接var connectState = this.Connect();if (!connectState.IsSuccessed){throw new Exception(connectState.Message);}// 子命令(位/字)byte writeCode = (byte)(typeof(T) == typeof(bool) ? 0x01 : 0x00);// 起始地址  XY    直接翻译  100   00 01 00    D100  64 00 00List<byte> startBytes = this.StartToBytes(areaCode, startAddr);//计算数据类型的长度int typeLen = this.CalculatLength<T>();int count = values.Count;//计算数据的字节列表List<byte> datas = this.GetDataBytes<T>(values);List<byte> baseBytes = new List<byte>{0x50,0x00,this._netCode,// 可变,根据PLC的设置0xFF,//PLC编号,固定值0xFF,0x03,//目标模块IO编号,固定FF 03this._stationCode,// 可变,目标模块站号};//0x0E,0x00,  // 剩余字节长度List<byte> commandBytes = new List<byte> {0x0A,0x00,//超时时间0x01,0x14,// 成批写入writeCode,0x00,// 字操作startBytes[0],startBytes[1],startBytes[2],// 起始地址(byte)areaCode,// 区域代码 (byte)(typeLen*count%256),(byte)(typeLen*count/256%256), //长度};commandBytes.AddRange(datas);baseBytes.Add((byte)(commandBytes.Count % 256));baseBytes.Add((byte)(commandBytes.Count / 256 % 256));baseBytes.AddRange(commandBytes);socket.Send(baseBytes.ToArray());// 解析响应byte[] respBytes = new byte[11];socket.Receive(respBytes, 0, 11, SocketFlags.None);// 状态if ((respBytes[9] | respBytes[10]) != 0x00){throw new Exception("响应异常。" + respBytes[9].ToString() + respBytes[10].ToString());}}catch (Exception ex){result.IsSuccessed = false;result.Message = ex.Message;}return result;}#endregion#region 私有方法/// <summary>/// 地址解析/// </summary>/// <param name="address">地址字符串</param>/// <returns></returns>public Tuple<AreaCode, string> AnalysisAddress(string address){// 取两个字符string area = address.Substring(0, 2);if (!new string[] { "TN", "TS", "CS", "CN" }.Contains(area)){area = address.Substring(0, 1);}string start = address.Substring(area.Length);// 返回一个元组对象 return new Tuple<AreaCode, string>((AreaCode)Enum.Parse(typeof(AreaCode), area), start);}/// <summary>/// 发送报文/// </summary>/// <param name="reqBytes">字节列表</param>/// <param name="count"></param>/// <returns></returns>/// <exception cref="Exception"></exception>public override List<byte> Send(List<byte> reqBytes, int count){socket.Send(reqBytes.ToArray());// 解析byte[] respBytes = new byte[11];socket.Receive(respBytes, 0, 11, SocketFlags.None);// 状态if ((respBytes[9] | respBytes[10]) != 0x00){throw new Exception("响应异常。" + respBytes[9].ToString() + respBytes[10].ToString());}// 数据长度 int dataLen = BitConverter.ToUInt16(new byte[] { respBytes[7], respBytes[8] },0) - 2;  // -2 的意思去除响应代码(状态)byte[] dataBytes = new byte[dataLen];socket.Receive(dataBytes, 0, dataLen, SocketFlags.None);return new List<byte>(dataBytes);}#endregion#region plc控制/// <summary>/// PLC远程启动/// </summary>/// <returns></returns>public Result Run(){return PlcStatus(0x01, new List<byte> { 0x00, 0x00 });}/// <summary>/// PLC远程停止/// </summary>/// <returns></returns>public Result Stop(){return PlcStatus(0x02);}/// <summary>/// PLC运行状态/// </summary>/// <param name="cmdCode"></param>/// <param name="cmd"></param>/// <returns></returns>private Result PlcStatus(byte cmdCode, List<byte> cmd = null){Result result = new Result();try{var connectState = this.Connect();if (!connectState.IsSuccessed){throw new Exception(connectState.Message);}List<byte> commandBytes = new List<byte>{0x50,0x00,this._netCode,// 可变,根据PLC的设置0xFF,0xFF,0x03,this._stationCode,// 可变};//0x08,0x00,  // 剩余字节长度List<byte> cmdBytes = new List<byte> {0x0A,0x00,cmdCode,0x10,0x00,0x00,0x01,0x00,//模式};if (cmd != null){cmdBytes.AddRange(cmd);}commandBytes.Add((byte)(commandBytes.Count % 256));commandBytes.Add((byte)(commandBytes.Count / 256 % 256));commandBytes.AddRange(cmdBytes);socket.Send(commandBytes.ToArray());byte[] respBytes = new byte[11];socket.Receive(respBytes, 0, 11, SocketFlags.None);// 状态if ((respBytes[9] | respBytes[10]) != 0x00){throw new Exception("响应异常。" + respBytes[1].ToString());}}catch (Exception ex){result.IsSuccessed = false;result.Message = ex.Message;}return result;}#endregion}
}

 4、测试通讯库

1、启动MC服务器

2、利用通讯库读写数据

1、读取D区100开始的3个short数据

 

2、读取M区100开始的5个float数据

 

 

3、读取X区100开始的4个bool数据

 

4、写入M区200开始的2个short数据

 

5、写入D区200开始的5个float数据

 

 

3、完整代码

 /// <summary>/// 测试A-3E通讯库/// </summary>static void MCLibTestA3E(){A3E qNA3E = new A3E("192.168.1.7", 6000);#region 读数据//Console.WriteLine("读取D区100开始的3个short数据");//var result1 = qNA3E.Read<short>("D100", 3);//if (result1.IsSuccessed)//{//    result1.Datas.ForEach(d => Console.WriteLine(d));//}//else//{//    Console.WriteLine(result1.Message);//}//Console.WriteLine("读取M区100开始的5个float数据");//var result2 = qNA3E.Read<float>("M100", 5);//if (result2.IsSuccessed)//{//    result2.Datas.ForEach(d => Console.WriteLine(d));//}//else//{//    Console.WriteLine(result2.Message);//}//Console.WriteLine("读取X区100开始的4个bool数据");//var result3 = qNA3E.Read<bool>(AreaCode.X, "100", 4);//if (result3.IsSuccessed)//{//    result3.Datas.ForEach(d => Console.WriteLine(d));//}//else//{//    Console.WriteLine(result3.Message);//}#endregion#region 写数据Console.WriteLine("写入M区200开始的2个short数据");var result4 = qNA3E.Write<short>(new List<short> { -541, 982 }, "M200");if (result4.IsSuccessed){Console.WriteLine(result4.Message);}Console.WriteLine("写入D区200开始的5个float数据");var result5 = qNA3E.Write<float>(new List<float> { 111, 0, -8076, 13.67f, -985.325f }, "D200");if (result5.IsSuccessed){Console.WriteLine(result5.Message);}#endregion }

5、小结

原创真的不容易,走过路过不要错过,点赞关注收藏又圈粉,共同致富。

原创真的不容易,走过路过不要错过,点赞关注收藏又圈粉,共同致富。

原创真的不容易,走过路过不要错过,点赞关注收藏又圈粉,共同致富

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

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

相关文章

头疼管理 MySQL 数据库 Schema?开源工具大盘点!

MySQL 是世界上最流行的开源关系型数据库管理系统 (RDBMS)&#xff0c;但是对 MySQL 数据库做 schema 变更 (schema migration) 还是有点难搞的&#x1f605;。 本文中&#xff0c;我们盘点一些好用的针对 MySQL 的开源数据库 schema 迁移工具&#xff0c;简单聊一下它们提供的…

Jenkins CI/CD 流程

Jenkins CI/CD 流程 这张图稍微更形象一点&#xff0c;上线之前先把代码git到版本仓库&#xff0c;然后通过Jenkins将Java项目通过maven去构建&#xff0c;这是在非容器之前&#xff0c;典型的自动化的一个版本上线流程。那它有哪些问题呢&#xff1f; 如&#xff1a;它的测试环…

Stable Diffusion——常用插件安装与测试(一)

前言 随着Stable Diffusion不断演进&#xff0c;越来越多的开发者开始涉足插件开发。尽管网络上存在大量教程&#xff0c;但它们通常零散分布&#xff0c;逐个学习和查找非常耗时&#xff0c;使人感觉每天都在劳累思考。这里总结了Stable Diffusion常用的插件安装与测试方法。…

Linux篇:开发工具yum/vim/gcc/g++/Makefile/gdb

一. yum&#xff1a;软件包管理器 什么是软件包&#xff1f; 在Linux 下安装软件 , 一个通常的办法是下载到程序的源代码 , 并进行编译 , 得到可执行程序 . 但是这样太麻烦了, 于是有些人把一些常用的软件提前编译好 , 做成软件包 (可以理解成windows 上的安装程序) 放在…

Tomcat 学习之 Servlet

目录 1 Servlet 介绍 2 创建一个 Servlet 3 web.xml 介绍&#xff08;不涉及 filter 和 listener 标签&#xff09; 3.1 display-name 3.2 welcome-file-list 3.3 servlet 3.4 session-config 3.5 error-page 4 Tomcat 如何根据 URL 定位到 Servlet 5 执行 Servlet …

Web基础②nginx搭建与配置

目录 一.Nginx概述 1.定义 2.Nginx模块作用 &#xff08;1&#xff09;main模块 &#xff08;2&#xff09;stream服务模块 &#xff08;3&#xff09;邮件服务模块 &#xff08;4&#xff09;第三方模块 &#xff08;5&#xff09;events模块 &#xff08;6&#xff…

STM32cubeMX配置FreeRTOS----互斥量

这篇文章为大家介绍FreeRTOS里的 互斥锁。 文章目录 前言一、互斥量的概念二、STM32cubeMX配置互斥量三、相关函数1. 创建互斥量2. 获取互斥量3. 释放互斥量 四、使用互斥量访问共享资源五、递归互斥锁在RTOS里互斥锁是谁获取&#xff0c;谁就释放吗&#xff1f; 总结 前言 一、…

MySQL引擎对决:深入解析MyISAM和InnoDB的区别

欢迎来到我的博客&#xff0c;代码的世界里&#xff0c;每一行都是一个故事 MySQL引擎对决&#xff1a;深入解析MyISAM和InnoDB的区别 前言引擎概述MyISAM&#xff1a;InnoDB&#xff1a; 使用场景使用 MyISAM 的最佳实践&#xff1a;使用 InnoDB 的最佳实践&#xff1a;可能的…

使用Windbg动态调试目标程序去分析异常的两实战案例分享

目录 1、前言 2、案例1&#xff1a;程序退出时弹出报错提示框 2.1、问题说明 2.2、到系统应用程序日志中看系统有没有自动生成dump文件 2.3、将Windbg附加到目标程序上进行动态调试 3、案例2&#xff1a;程序在运行过程中弹出ASSERT断言提示框 3.1、问题说明 3.2、将Wi…

Linux内核中并发与竞争的处理方法之原子操作简介

一. 简介 当我们发现驱动程序中存在并发和竞争的时候一定要处理掉&#xff0c;接下来我们依次来学习一下 Linux 内核提供的几种并发和竞争的处理方法。 本文学习Linux内核提供的一种处理并发与竞争的方法&#xff1a;原子操作。 二. 原子操作简介 原子操作就是指不能再进一步…

net反射

1.1 查找dll文件 Load需要把dll放到程序当前路径加载&#xff0c;也可以读取字符串形式。LoadFrom需要写全路径&#xff0c;如果test1.dll引用了test2.dll&#xff0c;同时也会加载test2.dll进来。LoadFile不会加载test2.dll。 Assembly assembly1 Assembly.Load("DllTe…

Gmail邮箱群发邮件的技巧?邮箱怎么绑定?

Gmail邮箱注册教程指南&#xff1f;如何注册新的Gmail邮箱帐户&#xff1f; Gmail邮箱作为谷歌推出的邮件服务&#xff0c;以其高效、稳定和便捷的特性受到广大用户的喜爱。然而&#xff0c;如何在Gmail中进行有效的群发邮件&#xff0c;接下来&#xff0c;蜂邮将给大家介绍一…