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

2024-02-19 07:28

本文主要是介绍C#上位机与三菱PLC的通信06--MC协议之QnA-3E报文测试,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1、A-3E报文回顾

1、存储区分类及访问规则 

2、命令类型

命令由主命令+子命令组成

 

3、报文结构

 2、启动mc服务器

3、创建VS项目

这节继续使用上节的VS2022的项目,增加一个方法 MCTestA3E(),具体怎么创建项目,见上节的过程。C#上位机与三菱PLC的通信04--MC协议之A-1E报文测试

 

4、报文组装与测试

 1、 读字数据,读取D100开始的2个数据, short/ushort

 

  /// <summary>/// A-3E报文测试/// </summary>private static void MCTestA3E(){// 连接Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);socket.Connect("192.168.1.7", 6000);#region 1、 读字数据,读取D100开始的2个数据, short/ushortbyte[] bytes = new byte[]{0x50,0x00,//请求副头部,固定50 000x00,// 网络号,可变,根据PLC的设置0xFF,//PLC编号,固定值0xFF,0x03,//目标模块IO编号,固定FF 030x00,// 可变,目标模块站号0x0C,0x00,  // 剩余字节长度,当前字节往后0x0A,0x00,//PLC响应超时时间,以250ms为单位计算0x01,0x04,// 成批读出,主命令0x00,0x00,// 字操作,子命令0x64,0x00,0x00,// 起始地址0xA8,// 区域代码 0x02,0x00 //读取长度     //  如果请求一个Float   2;2Float  4};socket.Send(bytes);// 暂时以这种方式来处理  byte[] respBytes = new byte[15];socket.Receive(respBytes);for (int i = 11; i < respBytes.Length; i++){// 小端处理,每2个字节作为一个数据byte[] dataBytes = new byte[2];dataBytes[0] = respBytes[i];dataBytes[1] = respBytes[++i];Console.WriteLine(BitConverter.ToInt16(dataBytes,0));}#endregion}

 2、 读字数据,读取D102开始的1个数据, float类型

 

 /// <summary>/// A-3E报文测试/// </summary>private static void MCTestA3E(){// 连接Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);socket.Connect("192.168.1.7", 6000);#region 1、 读字数据,读取D100开始的2个数据, short/ushort//byte[] bytes = new byte[]//{//    0x50,0x00,//请求副头部,固定50 00//    0x00,// 网络号,可变,根据PLC的设置//    0xFF,//PLC编号,固定值//    0xFF,0x03,//目标模块IO编号,固定FF 03//    0x00,// 可变,目标模块站号//    0x0C,0x00,  // 剩余字节长度,当前字节往后//    0x0A,0x00,//PLC响应超时时间,以250ms为单位计算//    0x01,0x04,// 成批读出,主命令//    0x00,0x00,// 字操作,子命令//    0x64,0x00,0x00,// 起始地址//    0xA8,// 区域代码 //    0x02,0x00 //读取长度     //  如果请求一个Float   2;2Float  4//};//socket.Send(bytes);暂时以这种方式来处理  //byte[] respBytes = new byte[15];//socket.Receive(respBytes);//for (int i = 11; i < respBytes.Length; i++)//{//    // 小端处理,每2个字节作为一个数据//    byte[] dataBytes = new byte[2];//    dataBytes[0] = respBytes[i];//    dataBytes[1] = respBytes[++i];//    Console.WriteLine(BitConverter.ToInt16(dataBytes,0));//}#endregion#region 2、 读字数据,读取D102开始的1个数据, float类型//byte[] bytes = new byte[]//{//    0x50,0x00,//请求副头部,固定50 00//    0x00,// 网络号,可变,根据PLC的设置//    0xFF,//PLC编号,固定值//    0xFF,0x03,//目标模块IO编号,固定FF 03//    0x00,// 可变,目标模块站号//    0x0C,0x00,  // 剩余字节长度,当前字节往后//    0x0A,0x00,//PLC响应超时时间,以250ms为单位计算//    0x01,0x04,// 成批读出,主命令//    0x00,0x00,// 字操作,子命令//    0x66,0x00,0x00,// 起始地址102//    0xA8,// 区域代码 //    0x02,0x00 //读取长度     //   请求一个Float占2个字节 //};//socket.Send(bytes);暂时以这种方式来处理  //byte[] respBytes = new byte[15];//socket.Receive(respBytes);//for (int i = 11; i < respBytes.Length; i++)//{//    //小端处理,每4个字节作为一组才是数据//    byte[] dataBytes = new byte[4];//    dataBytes[0] = respBytes[i];//    dataBytes[1] = respBytes[++i];//    dataBytes[2] = respBytes[++i];//    dataBytes[3] = respBytes[++i];//    Console.WriteLine(BitConverter.ToSingle(dataBytes,0));//字节转换成浮点数//}#endregion}

 3、 读位数据,即X102开始的1个位(true/false)

/// <summary>
/// A-3E报文测试
/// </summary>
private static void MCTestA3E()
{// 连接Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);socket.Connect("192.168.1.7", 6000);#region 1、 读字数据,读取D100开始的2个数据, short/ushort//byte[] bytes = new byte[]//{//    0x50,0x00,//请求副头部,固定50 00//    0x00,// 网络号,可变,根据PLC的设置//    0xFF,//PLC编号,固定值//    0xFF,0x03,//目标模块IO编号,固定FF 03//    0x00,// 可变,目标模块站号//    0x0C,0x00,  // 剩余字节长度,当前字节往后//    0x0A,0x00,//PLC响应超时时间,以250ms为单位计算//    0x01,0x04,// 成批读出,主命令//    0x00,0x00,// 字操作,子命令//    0x64,0x00,0x00,// 起始地址//    0xA8,// 区域代码 //    0x02,0x00 //读取长度     //  如果请求一个Float   2;2Float  4//};//socket.Send(bytes);暂时以这种方式来处理  //byte[] respBytes = new byte[15];//socket.Receive(respBytes);//for (int i = 11; i < respBytes.Length; i++)//{//    // 小端处理,每2个字节作为一个数据//    byte[] dataBytes = new byte[2];//    dataBytes[0] = respBytes[i];//    dataBytes[1] = respBytes[++i];//    Console.WriteLine(BitConverter.ToInt16(dataBytes,0));//}#endregion#region 2、 读字数据,读取D102开始的1个数据, float类型//byte[] bytes = new byte[]//{//    0x50,0x00,//请求副头部,固定50 00//    0x00,// 网络号,可变,根据PLC的设置//    0xFF,//PLC编号,固定值//    0xFF,0x03,//目标模块IO编号,固定FF 03//    0x00,// 可变,目标模块站号//    0x0C,0x00,  // 剩余字节长度,当前字节往后//    0x0A,0x00,//PLC响应超时时间,以250ms为单位计算//    0x01,0x04,// 成批读出,主命令//    0x00,0x00,// 字操作,子命令//    0x66,0x00,0x00,// 起始地址102//    0xA8,// 区域代码 //    0x02,0x00 //读取长度     //   请求一个Float占2个字节 //};//socket.Send(bytes);暂时以这种方式来处理  //byte[] respBytes = new byte[15];//socket.Receive(respBytes);//for (int i = 11; i < respBytes.Length; i++)//{//    //小端处理,每4个字节作为一组才是数据//    byte[] dataBytes = new byte[4];//    dataBytes[0] = respBytes[i];//    dataBytes[1] = respBytes[++i];//    dataBytes[2] = respBytes[++i];//    dataBytes[3] = respBytes[++i];//    Console.WriteLine(BitConverter.ToSingle(dataBytes,0));//字节转换成浮点数//}#endregion#region 3、 读位数据,即X102开始的1个位(true/false)byte[] bytes = new byte[]{0x50,0x00,//请求副头部,固定50 000x00,// 可变,根据PLC的设置0xFF,//PLC编号,固定值0xFF,0x03,//目标模块IO编号,固定FF 030x00,// 可变 ,目标模块站号0x0C,0x00,  // 剩余字节长度0x0A,0x00, //PLC响应超时时间,以250ms为单位计算0x01,0x4,// 成批读出 ,主命令0x01,0x00,// 子命令 - 位操作 0x02,0x01,0x00,// 起始地址,占3个字节,地址是102,用000102表示,因为是小端,前后颠倒,变成了0201000x9C,// 区域代码   X元件就是9C0x01,0x00 //读取长度     };socket.Send(bytes);byte[] respBytes = new byte[12];socket.Receive(respBytes);var obj = respBytes;string binaryStr = Convert.ToString(respBytes[11], 2).PadLeft(8, '0');//左移8位List<string> tempList = new List<string>();// 每转换一次可以拿两个位信息tempList.Add(binaryStr.Substring(0, 4));tempList.Add(binaryStr.Substring(4));for (int i = 0; i < 1; i++){Console.WriteLine(tempList[i] == "0001");}#endregion}

 4、 读字数据,即读取X100地址的int16类型数据,short或ushort类型的,如-79,35

 /// <summary>/// A-3E报文测试/// </summary>private static void MCTestA3E(){// 连接Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);socket.Connect("192.168.1.7", 6000);#region 1、 读字数据,读取D100开始的2个数据, short/ushort//byte[] bytes = new byte[]//{//    0x50,0x00,//请求副头部,固定50 00//    0x00,// 网络号,可变,根据PLC的设置//    0xFF,//PLC编号,固定值//    0xFF,0x03,//目标模块IO编号,固定FF 03//    0x00,// 可变,目标模块站号//    0x0C,0x00,  // 剩余字节长度,当前字节往后//    0x0A,0x00,//PLC响应超时时间,以250ms为单位计算//    0x01,0x04,// 成批读出,主命令//    0x00,0x00,// 字操作,子命令//    0x64,0x00,0x00,// 起始地址//    0xA8,// 区域代码 //    0x02,0x00 //读取长度     //  如果请求一个Float   2;2Float  4//};//socket.Send(bytes);暂时以这种方式来处理  //byte[] respBytes = new byte[15];//socket.Receive(respBytes);//for (int i = 11; i < respBytes.Length; i++)//{//    // 小端处理,每2个字节作为一个数据//    byte[] dataBytes = new byte[2];//    dataBytes[0] = respBytes[i];//    dataBytes[1] = respBytes[++i];//    Console.WriteLine(BitConverter.ToInt16(dataBytes,0));//}#endregion#region 2、 读字数据,读取D102开始的1个数据, float类型//byte[] bytes = new byte[]//{//    0x50,0x00,//请求副头部,固定50 00//    0x00,// 网络号,可变,根据PLC的设置//    0xFF,//PLC编号,固定值//    0xFF,0x03,//目标模块IO编号,固定FF 03//    0x00,// 可变,目标模块站号//    0x0C,0x00,  // 剩余字节长度,当前字节往后//    0x0A,0x00,//PLC响应超时时间,以250ms为单位计算//    0x01,0x04,// 成批读出,主命令//    0x00,0x00,// 字操作,子命令//    0x66,0x00,0x00,// 起始地址102//    0xA8,// 区域代码 //    0x02,0x00 //读取长度     //   请求一个Float占2个字节 //};//socket.Send(bytes);暂时以这种方式来处理  //byte[] respBytes = new byte[15];//socket.Receive(respBytes);//for (int i = 11; i < respBytes.Length; i++)//{//    //小端处理,每4个字节作为一组才是数据//    byte[] dataBytes = new byte[4];//    dataBytes[0] = respBytes[i];//    dataBytes[1] = respBytes[++i];//    dataBytes[2] = respBytes[++i];//    dataBytes[3] = respBytes[++i];//    Console.WriteLine(BitConverter.ToSingle(dataBytes,0));//字节转换成浮点数//}#endregion#region 3、 读位数据,即X102开始的1个位(true/false)//byte[] bytes = new byte[]//{//    0x50,0x00,//请求副头部,固定50 00//    0x00,// 可变,根据PLC的设置//    0xFF,//PLC编号,固定值//    0xFF,0x03,//目标模块IO编号,固定FF 03//    0x00,// 可变 ,目标模块站号//    0x0C,0x00,  // 剩余字节长度//    0x0A,0x00, //PLC响应超时时间,以250ms为单位计算//    0x01,0x4,// 成批读出 ,主命令//    0x01,0x00,// 子命令 - 位操作 //    0x02,0x01,0x00,// 起始地址,占3个字节,地址是102,用000102表示,因为是小端,前后颠倒,变成了020100//    0x9C,// 区域代码   X元件就是9C//    0x01,0x00 //读取长度     //};//socket.Send(bytes);//byte[] respBytes = new byte[12];//socket.Receive(respBytes);//var obj = respBytes;//string binaryStr = Convert.ToString(respBytes[11], 2).PadLeft(8, '0');//左移8位//List<string> tempList = new List<string>();每转换一次可以拿两个位信息//tempList.Add(binaryStr.Substring(0, 4));//tempList.Add(binaryStr.Substring(4));//for (int i = 0; i < 1; i++)//{//    Console.WriteLine(tempList[i] == "0001");//}#endregion#region 4、 读字数据,即读取X100地址的int16类型数据,short或ushort类型的,如-79,35byte[] bytes = new byte[]{0x50,0x00,0x00,// 可变,根据PLC的设置0xFF,0xFF,0x03,0x00,// 可变 0x0C,0x00,  // 剩余字节长度0x0A,0x00, 0x01,0x4,// 成批读出,主操作命令0x00,0x00,// 子命令 - 字操作 0x00,0x01,0x00,// 起始地址,100用3个字节表示就是000100,小端处理前后倒置就是0001000x9C,// 区域代码   X0x01,0x00 //读取长度     };socket.Send(bytes);byte[] respBytes = new byte[13];socket.Receive(respBytes);var obj = respBytes;for (int i = 11; i < respBytes.Length; i++){// 小端处理byte[] dataBytes = new byte[2];dataBytes[0] = respBytes[i];dataBytes[1] = respBytes[++i];Console.WriteLine(BitConverter.ToInt16(dataBytes,0));}#endregion}

5、写入字数据,即按字写入地址为D100开始的2个数据short类型的,即111/222

  /// <summary>/// A-3E报文测试/// </summary>private static void MCTestA3E(){// 连接Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);socket.Connect("192.168.1.7", 6000);#region 1、 读字数据,读取D100开始的2个数据, short/ushort//byte[] bytes = new byte[]//{//    0x50,0x00,//请求副头部,固定50 00//    0x00,// 网络号,可变,根据PLC的设置//    0xFF,//PLC编号,固定值//    0xFF,0x03,//目标模块IO编号,固定FF 03//    0x00,// 可变,目标模块站号//    0x0C,0x00,  // 剩余字节长度,当前字节往后//    0x0A,0x00,//PLC响应超时时间,以250ms为单位计算//    0x01,0x04,// 成批读出,主命令//    0x00,0x00,// 字操作,子命令//    0x64,0x00,0x00,// 起始地址//    0xA8,// 区域代码 //    0x02,0x00 //读取长度     //  如果请求一个Float   2;2Float  4//};//socket.Send(bytes);暂时以这种方式来处理  //byte[] respBytes = new byte[15];//socket.Receive(respBytes);//for (int i = 11; i < respBytes.Length; i++)//{//    // 小端处理,每2个字节作为一个数据//    byte[] dataBytes = new byte[2];//    dataBytes[0] = respBytes[i];//    dataBytes[1] = respBytes[++i];//    Console.WriteLine(BitConverter.ToInt16(dataBytes,0));//}#endregion#region 2、 读字数据,读取D102开始的1个数据, float类型//byte[] bytes = new byte[]//{//    0x50,0x00,//请求副头部,固定50 00//    0x00,// 网络号,可变,根据PLC的设置//    0xFF,//PLC编号,固定值//    0xFF,0x03,//目标模块IO编号,固定FF 03//    0x00,// 可变,目标模块站号//    0x0C,0x00,  // 剩余字节长度,当前字节往后//    0x0A,0x00,//PLC响应超时时间,以250ms为单位计算//    0x01,0x04,// 成批读出,主命令//    0x00,0x00,// 字操作,子命令//    0x66,0x00,0x00,// 起始地址102//    0xA8,// 区域代码 //    0x02,0x00 //读取长度     //   请求一个Float占2个字节 //};//socket.Send(bytes);暂时以这种方式来处理  //byte[] respBytes = new byte[15];//socket.Receive(respBytes);//for (int i = 11; i < respBytes.Length; i++)//{//    //小端处理,每4个字节作为一组才是数据//    byte[] dataBytes = new byte[4];//    dataBytes[0] = respBytes[i];//    dataBytes[1] = respBytes[++i];//    dataBytes[2] = respBytes[++i];//    dataBytes[3] = respBytes[++i];//    Console.WriteLine(BitConverter.ToSingle(dataBytes,0));//字节转换成浮点数//}#endregion#region 3、 读位数据,即X102开始的1个位(true/false)//byte[] bytes = new byte[]//{//    0x50,0x00,//请求副头部,固定50 00//    0x00,// 可变,根据PLC的设置//    0xFF,//PLC编号,固定值//    0xFF,0x03,//目标模块IO编号,固定FF 03//    0x00,// 可变 ,目标模块站号//    0x0C,0x00,  // 剩余字节长度//    0x0A,0x00, //PLC响应超时时间,以250ms为单位计算//    0x01,0x4,// 成批读出 ,主命令//    0x01,0x00,// 子命令 - 位操作 //    0x02,0x01,0x00,// 起始地址,占3个字节,地址是102,用000102表示,因为是小端,前后颠倒,变成了020100//    0x9C,// 区域代码   X元件就是9C//    0x01,0x00 //读取长度     //};//socket.Send(bytes);//byte[] respBytes = new byte[12];//socket.Receive(respBytes);//var obj = respBytes;//string binaryStr = Convert.ToString(respBytes[11], 2).PadLeft(8, '0');//左移8位//List<string> tempList = new List<string>();每转换一次可以拿两个位信息//tempList.Add(binaryStr.Substring(0, 4));//tempList.Add(binaryStr.Substring(4));//for (int i = 0; i < 1; i++)//{//    Console.WriteLine(tempList[i] == "0001");//}#endregion#region 4、 读字数据,即读取X100地址的int16类型数据,short或ushort类型的,如-79,35//byte[] bytes = new byte[]//{//    0x50,0x00,//    0x00,// 可变,根据PLC的设置//    0xFF,//    0xFF,0x03,//    0x00,// 可变 //    0x0C,0x00,  // 剩余字节长度//    0x0A,0x00, //    0x01,0x4,// 成批读出,主操作命令//    0x00,0x00,// 子命令 - 字操作 //    0x00,0x01,0x00,// 起始地址,100用3个字节表示就是000100,小端处理前后倒置就是000100//    0x9C,// 区域代码   X//    0x01,0x00 //读取长度     //};//socket.Send(bytes);//byte[] respBytes = new byte[13];//socket.Receive(respBytes);//var obj = respBytes;//for (int i = 11; i < respBytes.Length; i++)//{//    // 小端处理//    byte[] dataBytes = new byte[2];//    dataBytes[0] = respBytes[i];//    dataBytes[1] = respBytes[++i];//    Console.WriteLine(BitConverter.ToInt16(dataBytes,0));//}#endregion#region 5、写入字数据,即按字写入地址为D100开始的2个数据short类型的,即111/222byte[] bytes = new byte[]{0x50,0x00,0x00,// 可变,根据PLC的设置0xFF,0xFF,0x03,0x00,// 可变 0x10,0x00,  // 剩余字节长度0x0A,0x00, 0x01,0x14,// 成批写入0x00,0x00,// 字操作 0x64,0x00,0x00,// 起始地址0xA8,// 区域代码 0x02,0x00, //写入长度//具体的数据值,111转换成16进制就是6f,小端处理就是6f00,222小端处理就是de00(byte)(111%256),//低位(byte)(111/256%256),//高位(byte)(222%256),(byte)(222/256%256)};socket.Send(bytes);byte[] respBytes = new byte[11];socket.Receive(respBytes);var obj = respBytes;//最后2个字节是状态码,0X00,0X00是状态码,如果是0,表示成功for (int i = 9; i < respBytes.Length; i++){// 小端处理,每2个字节作为一个数据byte[] dataBytes = new byte[2];dataBytes[0] = respBytes[i];dataBytes[1] = respBytes[++i];if (BitConverter.ToInt16(dataBytes,0) == 0){Console.WriteLine("写入成功");}}#endregion}

 6、写入字数据,即按字写入地址为D102开始的1个数据,即float类型的13.9

 /// <summary>/// A-3E报文测试/// </summary>private static void MCTestA3E(){// 连接Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);socket.Connect("192.168.1.7", 6000);#region 1、 读字数据,读取D100开始的2个数据, short/ushort//byte[] bytes = new byte[]//{//    0x50,0x00,//请求副头部,固定50 00//    0x00,// 网络号,可变,根据PLC的设置//    0xFF,//PLC编号,固定值//    0xFF,0x03,//目标模块IO编号,固定FF 03//    0x00,// 可变,目标模块站号//    0x0C,0x00,  // 剩余字节长度,当前字节往后//    0x0A,0x00,//PLC响应超时时间,以250ms为单位计算//    0x01,0x04,// 成批读出,主命令//    0x00,0x00,// 字操作,子命令//    0x64,0x00,0x00,// 起始地址//    0xA8,// 区域代码 //    0x02,0x00 //读取长度     //  如果请求一个Float   2;2Float  4//};//socket.Send(bytes);暂时以这种方式来处理  //byte[] respBytes = new byte[15];//socket.Receive(respBytes);//for (int i = 11; i < respBytes.Length; i++)//{//    // 小端处理,每2个字节作为一个数据//    byte[] dataBytes = new byte[2];//    dataBytes[0] = respBytes[i];//    dataBytes[1] = respBytes[++i];//    Console.WriteLine(BitConverter.ToInt16(dataBytes,0));//}#endregion#region 2、 读字数据,读取D102开始的1个数据, float类型//byte[] bytes = new byte[]//{//    0x50,0x00,//请求副头部,固定50 00//    0x00,// 网络号,可变,根据PLC的设置//    0xFF,//PLC编号,固定值//    0xFF,0x03,//目标模块IO编号,固定FF 03//    0x00,// 可变,目标模块站号//    0x0C,0x00,  // 剩余字节长度,当前字节往后//    0x0A,0x00,//PLC响应超时时间,以250ms为单位计算//    0x01,0x04,// 成批读出,主命令//    0x00,0x00,// 字操作,子命令//    0x66,0x00,0x00,// 起始地址102//    0xA8,// 区域代码 //    0x02,0x00 //读取长度     //   请求一个Float占2个字节 //};//socket.Send(bytes);暂时以这种方式来处理  //byte[] respBytes = new byte[15];//socket.Receive(respBytes);//for (int i = 11; i < respBytes.Length; i++)//{//    //小端处理,每4个字节作为一组才是数据//    byte[] dataBytes = new byte[4];//    dataBytes[0] = respBytes[i];//    dataBytes[1] = respBytes[++i];//    dataBytes[2] = respBytes[++i];//    dataBytes[3] = respBytes[++i];//    Console.WriteLine(BitConverter.ToSingle(dataBytes,0));//字节转换成浮点数//}#endregion#region 3、 读位数据,即X102开始的1个位(true/false)//byte[] bytes = new byte[]//{//    0x50,0x00,//请求副头部,固定50 00//    0x00,// 可变,根据PLC的设置//    0xFF,//PLC编号,固定值//    0xFF,0x03,//目标模块IO编号,固定FF 03//    0x00,// 可变 ,目标模块站号//    0x0C,0x00,  // 剩余字节长度//    0x0A,0x00, //PLC响应超时时间,以250ms为单位计算//    0x01,0x4,// 成批读出 ,主命令//    0x01,0x00,// 子命令 - 位操作 //    0x02,0x01,0x00,// 起始地址,占3个字节,地址是102,用000102表示,因为是小端,前后颠倒,变成了020100//    0x9C,// 区域代码   X元件就是9C//    0x01,0x00 //读取长度     //};//socket.Send(bytes);//byte[] respBytes = new byte[12];//socket.Receive(respBytes);//var obj = respBytes;//string binaryStr = Convert.ToString(respBytes[11], 2).PadLeft(8, '0');//左移8位//List<string> tempList = new List<string>();每转换一次可以拿两个位信息//tempList.Add(binaryStr.Substring(0, 4));//tempList.Add(binaryStr.Substring(4));//for (int i = 0; i < 1; i++)//{//    Console.WriteLine(tempList[i] == "0001");//}#endregion#region 4、 读字数据,即读取X100地址的int16类型数据,short或ushort类型的,如-79,35//byte[] bytes = new byte[]//{//    0x50,0x00,//    0x00,// 可变,根据PLC的设置//    0xFF,//    0xFF,0x03,//    0x00,// 可变 //    0x0C,0x00,  // 剩余字节长度//    0x0A,0x00, //    0x01,0x4,// 成批读出,主操作命令//    0x00,0x00,// 子命令 - 字操作 //    0x00,0x01,0x00,// 起始地址,100用3个字节表示就是000100,小端处理前后倒置就是000100//    0x9C,// 区域代码   X//    0x01,0x00 //读取长度     //};//socket.Send(bytes);//byte[] respBytes = new byte[13];//socket.Receive(respBytes);//var obj = respBytes;//for (int i = 11; i < respBytes.Length; i++)//{//    // 小端处理//    byte[] dataBytes = new byte[2];//    dataBytes[0] = respBytes[i];//    dataBytes[1] = respBytes[++i];//    Console.WriteLine(BitConverter.ToInt16(dataBytes,0));//}#endregion#region 5、写入字数据,即按字写入地址为D100开始的2个数据short类型的,即111/222//byte[] bytes = new byte[]//{//    0x50,0x00,//    0x00,// 可变,根据PLC的设置//    0xFF,//    0xFF,0x03,//    0x00,// 可变 //    0x10,0x00,  // 剩余字节长度//    0x0A,0x00, //    0x01,0x14,// 成批写入//    0x00,0x00,// 字操作 //    0x64,0x00,0x00,// 起始地址//    0xA8,// 区域代码 //    0x02,0x00, //写入长度//    //具体的数据值,111转换成16进制就是6f,小端处理就是6f00,222小端处理就是de00//    (byte)(111%256),//低位//    (byte)(111/256%256),//高位//    (byte)(222%256),//    (byte)(222/256%256)//};//socket.Send(bytes);//byte[] respBytes = new byte[11];//socket.Receive(respBytes);//var obj = respBytes;最后2个字节是状态码,0X00,0X00是状态码,如果是0,表示成功//for (int i = 9; i < respBytes.Length; i++)//{//    // 小端处理,每2个字节作为一个数据//    byte[] dataBytes = new byte[2];//    dataBytes[0] = respBytes[i];//    dataBytes[1] = respBytes[++i];//    if (BitConverter.ToInt16(dataBytes,0) == 0)//    {//        Console.WriteLine("写入成功");//    }//}#endregion#region 6、写入字数据,即按字写入地址为D102开始的1个数据,即float类型的13.9float value = 13.9f;byte[] bytes = new byte[]{0x50,0x00,0x00,// 可变,根据PLC的设置0xFF,0xFF,0x03,0x00,// 可变 0x10,0x00,  // 剩余字节长度0x0A,0x00,0x01,0x14,// 成批写入0x00,0x00,// 字操作 0x66,0x00,0x00,// 起始地址,占3个字节,102转换成16进制的小端格式是6600000xA8,// 区域代码 0x02,0x00, //写入长度,float的长度是2个//具体的数据值,float占4个字节,分别是66,66,5e,41BitConverter.GetBytes(value)[0],BitConverter.GetBytes(value)[1],BitConverter.GetBytes(value)[2],BitConverter.GetBytes(value)[3]};socket.Send(bytes);byte[] respBytes = new byte[11];socket.Receive(respBytes);var obj = respBytes;//最后2个字节是状态码,0X00,0X00是状态码,如果是0,表示成功for (int i = 9; i < respBytes.Length; i++){// 小端处理,每2个字节作为一个数据byte[] dataBytes = new byte[2];dataBytes[0] = respBytes[i];dataBytes[1] = respBytes[++i];if (BitConverter.ToInt16(dataBytes,0) == 0){Console.WriteLine("写入成功");}}#endregion}

5、小结

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

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

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

 

这篇关于C#上位机与三菱PLC的通信06--MC协议之QnA-3E报文测试的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/723938

相关文章

C#中checked关键字的使用小结

《C#中checked关键字的使用小结》本文主要介绍了C#中checked关键字的使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录✅ 为什么需要checked? 问题:整数溢出是“静默China编程”的(默认)checked的三种用

C#中预处理器指令的使用小结

《C#中预处理器指令的使用小结》本文主要介绍了C#中预处理器指令的使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录 第 1 名:#if/#else/#elif/#endif✅用途:条件编译(绝对最常用!) 典型场景: 示例

C#实现将XML数据自动化地写入Excel文件

《C#实现将XML数据自动化地写入Excel文件》在现代企业级应用中,数据处理与报表生成是核心环节,本文将深入探讨如何利用C#和一款优秀的库,将XML数据自动化地写入Excel文件,有需要的小伙伴可以... 目录理解XML数据结构与Excel的对应关系引入高效工具:使用Spire.XLS for .NETC

C#如何在Excel文档中获取分页信息

《C#如何在Excel文档中获取分页信息》在日常工作中,我们经常需要处理大量的Excel数据,本文将深入探讨如何利用Spire.XLSfor.NET,高效准确地获取Excel文档中的分页信息,包括水平... 目录理解Excel中的分页机制借助 Spire.XLS for .NET 获取分页信息为什么选择 S

C#高效实现在Word文档中自动化创建图表的可视化方案

《C#高效实现在Word文档中自动化创建图表的可视化方案》本文将深入探讨如何利用C#,结合一款功能强大的第三方库,实现在Word文档中自动化创建图表,为你的数据呈现和报告生成提供一套实用且高效的解决方... 目录Word文档图表自动化:为什么选择C#?从零开始:C#实现Word文档图表的基本步骤深度优化:C

在C#中分离饼图的某个区域的操作指南

《在C#中分离饼图的某个区域的操作指南》在处理Excel饼图时,我们可能需要将饼图的各个部分分离出来,以使它们更加醒目,Spire.XLS提供了Series.DataFormat.Percent属性,... 目录引言如何设置饼图各分片之间分离宽度的代码示例:从整个饼图中分离单个分片的代码示例:引言在处理

C#借助Spire.XLS for .NET实现在Excel中添加文档属性

《C#借助Spire.XLSfor.NET实现在Excel中添加文档属性》在日常的数据处理和项目管理中,Excel文档扮演着举足轻重的角色,本文将深入探讨如何在C#中借助强大的第三方库Spire.... 目录为什么需要程序化添加Excel文档属性使用Spire.XLS for .NET库实现文档属性管理Sp

C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解

《C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解》:本文主要介绍C++,C#,Rust,Go,Java,Python,JavaScript性能对比全面... 目录编程语言性能对比、核心优势与最佳使用场景性能对比表格C++C#RustGoJavapythonjav

C# 预处理指令(# 指令)的具体使用

《C#预处理指令(#指令)的具体使用》本文主要介绍了C#预处理指令(#指令)的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录1、预处理指令的本质2、条件编译指令2.1 #define 和 #undef2.2 #if, #el

C#实现将Excel工作表拆分为多个窗格

《C#实现将Excel工作表拆分为多个窗格》在日常工作中,我们经常需要处理包含大量数据的Excel文件,本文将深入探讨如何在C#中利用强大的Spire.XLSfor.NET自动化实现Excel工作表的... 目录为什么需要拆分 Excel 窗格借助 Spire.XLS for .NET 实现冻结窗格(Fro