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# string转unicode字符的实现

《C#string转unicode字符的实现》本文主要介绍了C#string转unicode字符的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随... 目录1. 获取字符串中每个字符的 Unicode 值示例代码:输出:2. 将 Unicode 值格式化

SpringBoot中整合RabbitMQ(测试+部署上线最新完整)的过程

《SpringBoot中整合RabbitMQ(测试+部署上线最新完整)的过程》本文详细介绍了如何在虚拟机和宝塔面板中安装RabbitMQ,并使用Java代码实现消息的发送和接收,通过异步通讯,可以优化... 目录一、RabbitMQ安装二、启动RabbitMQ三、javascript编写Java代码1、引入

Qt 中集成mqtt协议的使用方法

《Qt中集成mqtt协议的使用方法》文章介绍了如何在工程中引入qmqtt库,并通过声明一个单例类来暴露订阅到的主题数据,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录一,引入qmqtt 库二,使用一,引入qmqtt 库我是将整个头文件/源文件都添加到了工程中进行编译,这样 跨平台

Nginx设置连接超时并进行测试的方法步骤

《Nginx设置连接超时并进行测试的方法步骤》在高并发场景下,如果客户端与服务器的连接长时间未响应,会占用大量的系统资源,影响其他正常请求的处理效率,为了解决这个问题,可以通过设置Nginx的连接... 目录设置连接超时目的操作步骤测试连接超时测试方法:总结:设置连接超时目的设置客户端与服务器之间的连接

C#中读取XML文件的四种常用方法

《C#中读取XML文件的四种常用方法》Xml是Internet环境中跨平台的,依赖于内容的技术,是当前处理结构化文档信息的有力工具,下面我们就来看看C#中读取XML文件的方法都有哪些吧... 目录XML简介格式C#读取XML文件方法使用XmlDocument使用XmlTextReader/XmlTextWr

C#比较两个List集合内容是否相同的几种方法

《C#比较两个List集合内容是否相同的几种方法》本文详细介绍了在C#中比较两个List集合内容是否相同的方法,包括非自定义类和自定义类的元素比较,对于非自定义类,可以使用SequenceEqual、... 目录 一、非自定义类的元素比较1. 使用 SequenceEqual 方法(顺序和内容都相等)2.

C#使用DeepSeek API实现自然语言处理,文本分类和情感分析

《C#使用DeepSeekAPI实现自然语言处理,文本分类和情感分析》在C#中使用DeepSeekAPI可以实现多种功能,例如自然语言处理、文本分类、情感分析等,本文主要为大家介绍了具体实现步骤,... 目录准备工作文本生成文本分类问答系统代码生成翻译功能文本摘要文本校对图像描述生成总结在C#中使用Deep

C#从XmlDocument提取完整字符串的方法

《C#从XmlDocument提取完整字符串的方法》文章介绍了两种生成格式化XML字符串的方法,方法一使用`XmlDocument`的`OuterXml`属性,但输出的XML字符串不带格式,可读性差,... 方法1:通过XMLDocument的OuterXml属性,见XmlDocument类该方法获得的xm

C#多线程编程中导致死锁的常见陷阱和避免方法

《C#多线程编程中导致死锁的常见陷阱和避免方法》在C#多线程编程中,死锁(Deadlock)是一种常见的、令人头疼的错误,死锁通常发生在多个线程试图获取多个资源的锁时,导致相互等待对方释放资源,最终形... 目录引言1. 什么是死锁?死锁的典型条件:2. 导致死锁的常见原因2.1 锁的顺序问题错误示例:不同

C#提取PDF表单数据的实现流程

《C#提取PDF表单数据的实现流程》PDF表单是一种常见的数据收集工具,广泛应用于调查问卷、业务合同等场景,凭借出色的跨平台兼容性和标准化特点,PDF表单在各行各业中得到了广泛应用,本文将探讨如何使用... 目录引言使用工具C# 提取多个PDF表单域的数据C# 提取特定PDF表单域的数据引言PDF表单是一