解析西门子PLC的String和WString

西门子PLC有两种字符串类型,String与WString

String 用于存放英文数字标点符号等ASCII字符,每个字符占用一个字节

WString宽字符串用于存放中文、英文、数字等Unicode字符,每个字符占用两个字节

之前我搞过一篇解析String的

关于使用TCP-S7协议读写西门子PLC字符串的问题_s7 写入string-CSDN博客

对西门子PLC字符串的读写逻辑如下:

String[50]为例:


西门子PLC字符串类型 String[50],占用52个字节(偏移量),第一个字节是最大长度,就是50,第二个字节是实际长度24,第三个字节之后就是ASCII码,

PLC解析逻辑为找到第二个字节的长度length.,然后查找length个字符,就结束

WString[50]为例:

西门子PLC字符串类型 WString[50],占用104个字节(偏移量),第一第二个字节是最大长度,就是50,第三第四个字节是实际长度10,第五个字节之后就是Unicode码,

PLC解析逻辑为找到第三第四个字节的长度actualLength,然后查找actualLength * 2个字节,就结束。然后将字节数组转换为Unicode字符串即可 ,使用方法Encoding.Unicode.GetString(buffer, 4, actualLength * 2),因C#是低字节在前的,因此需要将相邻两个元素交换 buffer[0]与buffer[1]交换,buffer[2]与buffer[3]交换

相关解析函数如下【编程语言:C#】

​/// <summary>/// 写入西门子PLC的宽字符串,写入的字节长度为(length+2)*2,其中第一第二字节代表最大长度【PLC设定的】,第三第四字节代表实际长度/// 从第五个字节开始Unicode编码/// </summary>/// <param name="dbNumber"></param>/// <param name="offsetAddress"></param>/// <param name="maxLength"></param>/// <param name="writeString"></param>/// <returns></returns>static bool DB_WriteWString(short dbNumber, ushort offsetAddress, int maxLength, string writeString){byte[] byteArray = new byte[(maxLength + 2) * 2];byte[] stringByteArray = System.Text.Encoding.Unicode.GetBytes(writeString);//因C#是低字节在前的,因此需要交换相邻两个元素的位置。for (int i = 0; i < stringByteArray.Length / 2; i++){byte tempElement = stringByteArray[2 * i];stringByteArray[2 * i] = stringByteArray[2 * i + 1];stringByteArray[2 * i + 1] = tempElement;}Array.Copy(BitConverter.GetBytes((ushort)maxLength).Reverse().ToArray(), 0, byteArray, 0, 2);//最大长度Array.Copy(BitConverter.GetBytes((ushort)writeString.Length).Reverse().ToArray(), 0, byteArray, 2, 2);//实际长度Array.Copy(stringByteArray, 0, byteArray, 4, stringByteArray.Length);//第三个字节开始写入字符串的ASCII码return WriteSerialByteArray(dbNumber, offsetAddress, byteArray);}static bool WriteSerialByteArray(short dbNumber, ushort offsetAddress, byte[] byteArray){Console.WriteLine($"这里写入DB{dbNumber}.{offsetAddress}的连续字节数组【{string.Join(",", byteArray.Select(x => x.ToString("X2")))}】");return true;}static bool ReadSerialByteArray(short dbNumber, ushort offsetAddress, out byte[] byteArray){byteArray = new byte[] { 12, 6, 48, 49, 97, 98, 65, 66, 0, 0, 0, 0 };//假设读取12个字节Console.WriteLine($"这里读取DB{dbNumber}.{offsetAddress}的连续字节数组【{string.Join(",", byteArray.Select(x => x.ToString("X2")))}】");return true;}/// <summary>/// 读取宽字符串Unicode:从第五个字节开始,读取实际长度个字符  actualLength =  byteArray[2] byteArray[3]/// </summary>/// <param name="dbNumber"></param>/// <param name="offsetAddress"></param>/// <param name="readString"></param>/// <returns></returns>static bool DB_ReadWString(short dbNumber, ushort offsetAddress, out string readString){ReadSerialByteArray(dbNumber, offsetAddress, out byte[] byteArray);if (byteArray.Length < 4){throw new Exception($"读取宽字符串时,原字节数组长度不能低于4");}int maxLength = BitConverter.ToUInt16(new byte[] { byteArray[1], byteArray[0] }, 0);//最大长度,限制长度int actualLength = BitConverter.ToUInt16(new byte[] { byteArray[3], byteArray[2] }, 0);//实际长度if (byteArray.Length / 2 - 2 < actualLength){throw new Exception($"读取到的字节长度 小于 实际长度,非法的读取,PLC设定的字符串最大长度为【{maxLength}】");}//Unicode需要相邻两个字节交换位置,C#是低字节在前的for (int exchangeIndex = 0; exchangeIndex < actualLength; exchangeIndex++){byte tempExchange = byteArray[4 + exchangeIndex * 2];byteArray[4 + exchangeIndex * 2] = byteArray[4 + exchangeIndex * 2 + 1];byteArray[4 + exchangeIndex * 2 + 1] = tempExchange;}//从第五个元素开始,读取【actualLength * 2】个字节,一个Unicode字符对应一个ushort数字,占用两个字节readString = System.Text.Encoding.Unicode.GetString(byteArray, 4, actualLength * 2);return true;}​

假设PLC定义一个WString[50],我们读取到的字节数组为:

byte[] byteArray = new byte[] { 0, 50, 0, 16, 83, 228, 82, 81, 89, 71, 140, 45, 0, 65, 0, 66, 0, 67, 0, 49, 0, 50, 0, 51, 0, 52, 0, 53, 0, 54, 0, 97, 0, 98, 0, 99, 0, 0, 0, 0};

解析为:古剑奇谭ABC123456abc

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

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

相关文章

nginx基础使用

文章目录 nginx下载和编译configtest1test2config 原理 nginx 功能: 做为web server 使用在局域网内&#xff0c;提供对外的ip和端口 下载和编译 源码内容&#xff1a; nginx openssl pcrc zlib 编译&#xff1a; 1 cmake 方式&#xff1a; mkdir build cd build cmake 2 ma…

Unity Shader动画:用代码绘制动态视觉效果

在Unity中&#xff0c;Shader是运行在GPU上的小程序&#xff0c;用于控制顶点和像素的渲染过程。通过编写自定义Shader&#xff0c;开发者可以创造出各种令人惊叹的动画效果&#xff0c;从简单的颜色变化到复杂的流体模拟。本文将探讨如何使用Unity Shader来实现动画效果。 Sh…

算法入门篇(五)之 树的应用

目录 1.树和二叉树 1.1树&#xff08;Tree&#xff09; 1.1.1 特点 1.1.2 使用场景 1.1.3 示例 1.2二叉树&#xff08;Binary Tree&#xff09; 1.2.1 特点 1.2.2 使用场景 1.2.3 示例 2.二叉树遍历 2.1 先序遍历、中序遍历、后序遍历、层次遍历 2.1.1 先序遍历&…

git命令实现github与gitee同步

使用 git remote -v查看远程库连接了啥 git remote set-url --add origin 你的git仓库ssh (意思就是在 远端库origin下面加一个)然后就是git push&#xff08;这里可能会碰到问题&#xff0c;远程仓库的分支比本地分支更新&#xff09;注意github&#xff08;main&#xff09;与…

Vue3 Pinia的创建与使用代替Vuex 全局数据共享 同步异步

介绍 提供跨组件和页面的共享状态能力&#xff0c;作为Vuex的替代品&#xff0c;专为Vue3设计的状态管理库。 Vuex&#xff1a;在Vuex中&#xff0c;更改状态必须通过Mutation或Action完成&#xff0c;手动触发更新。Pinia&#xff1a;Pinia的状态是响应式的&#xff0c;当状…

Linux内核 mmap内存映射的实现原理

在Linux内核以及Linux系统编程的时候&#xff0c;经常会碰到mmap内存映射&#xff0c;mmap函数是实现高性能编程的一个关键点。本文详细介绍一下mmap实现原理。 虚拟地址映射物理地址 虚拟地址映射物理地址采用的是页表机制&#xff0c;64位CPU采用的是4级页表。 64位CPU虚拟…

鸿蒙 HarmonyOS NEXT端云一体化开发-认证服务篇

一、开通认证服务 地址&#xff1a;AppGallery Connect (huawei.com) 步骤&#xff1a; 1 进入到项目设置页面中&#xff0c;并点击左侧菜单中的认证服务 2 选择需要开通的服务并开通二、端侧项目环境配置 添加依赖 entry目录下的oh-package.json5 // 添加&#xff1a;主要前…

《python程序语言设计》第6章14题 估算派值 类似莱布尼茨函数。但是我看不明白

这个题提供的公式我没看明白&#xff0c;后来在网上找到了莱布尼茨函数 c 0 for i in range(1, 902, 100):a (-1) ** (i 1)b 2 * i - 1c a / bprint(i, round(4 / c, 3))结果 #按题里的信息&#xff0c;但是结果不对&#xff0c;莱布尼茨函数到底怎么算呀。

本地部署大模型

模型排行榜&#xff1a;https://www.superclueai.com/ Open WebUI https://docs.openwebui.com/ Open WebUI 是一种可扩展、功能丰富且用户友好的自托管 WebUI&#xff0c;旨在完全离线运行。它支持各种 LLM 运行器&#xff0c;包括 Ollama 和 OpenAI 兼容的 API。 docker安装…

PyTorch深度学习快速入门(上)

PyTorch深度学习快速入门&#xff08;上&#xff09; 一、前言&#xff08;一&#xff09;PyTorch环境配置&#xff08;二&#xff09;Python编译器的选择&#xff08;三&#xff09;Python学习中的两大法宝函数 二、如何加载数据&#xff08;一&#xff09;Dataset与Dataloade…

c++实现一个函数,对一个输入的整数,代表时间秒数,将其转换成时间格式字符串,如“01:09:11“,代表1小时,09分,11秒

c实现一个函数&#xff0c;对一个输入的整数&#xff0c;代表时间秒数&#xff0c;将其转换成时间格式字符串&#xff0c;如“01&#xff1a;09&#xff1a;11“&#xff0c;代表1小时&#xff0c;09分&#xff0c;11秒。 #include <iostream> #include <string> …

轻松学EntityFramework Core--模型创建

一、使用代码优先&#xff08;Code-First&#xff09;创建模型 Code-First 方法是 EF Core 提供的一种用于定义模型的方式&#xff0c;它允许开发人员通过编写 C# 类来定义数据库模式&#xff0c;再通过迁移命令生成数据库表。下面我们来一起看一下代码优先如何使用。 1.1、创…

lua 游戏架构 之 游戏 AI (六)ai_auto_skill

定义一个为ai_auto_skill的类&#xff0c;继承自ai_base类。ai_auto_skill类的目的是在AI自动战斗模式下&#xff0c;根据配置和条件自动选择并使用技能。 lua 游戏架构 之 游戏 AI &#xff08;一&#xff09;ai_base-CSDN博客文章浏览阅读379次。定义了一套接口和属性&#…

【原创】使用keepalived虚拟IP(VIP)实现MySQL的高可用故障转移

1. 背景 A、B服务器均部署有MySQL数据库&#xff0c;且互为主主。此处为A、B服务器部署MySQL数据库实现高可用的部署&#xff0c;当其中一台MySQL宕机后&#xff0c;VIP可自动切换至另一台MySQL提供服务&#xff0c;实现故障的自动迁移&#xff0c;实现高可用的目的。具体流程…

快速安装torch-gpu和Tensorflow-gpu(自用,Ubuntu)

要更详细的教程可以参考Tensorflow PyTorch 安装&#xff08;CPU GPU 版本&#xff09;&#xff0c;这里是有基础之后的快速安装。 一、Pytorch 安装 conda create -n torch_env python3.10.13 conda activate torch_env conda install cudatoolkit11.8 -c nvidia pip ins…

HarmonyOS NEXT 开发之ArkTS基础入门

ArkTS 是 HarmonyOS NEXT 的开发语言&#xff0c;它基于 TypeScript 并进行了扩展和优化。以下是一些基础语法知识点、示例用法及注意事项。 一、ArkTS 简介 ArkTS 是一种基于 TypeScript 的编程语言&#xff0c;主要用于 HarmonyOS 应用的 UI 界面和业务逻辑开发。它在 Type…

Android CTS兼容性测试工具介绍

参考官方文档&#xff1a;兼容性测试套件 | Android Open Source Project 参考文章&#xff1a;Android --- 一篇带你搞懂CTS_android cts-CSDN博客

mstc远程连接不锁屏

连接不锁屏 方法一 方法二 win10 解决多用户同时远程连接教程&#xff08;超详细图文&#xff09;_win10多用户登录-CSDN博客 win7软件 logout.bat for /f "skip1 tokens3" %%s in (query user %USERNAME%) do (%windir%\System32\tscon.exe %%s /dest:console) …

C# 抽象工厂模式

栏目总目录 概念 抽象工厂模式是一种创建型设计模式&#xff0c;它提供了一种创建一系列相关或相互依赖对象的接口&#xff0c;而无需指定它们具体的类。在抽象工厂模式中&#xff0c;一个抽象的工厂类负责定义创建产品对象的接口&#xff0c;但是具体工厂类将负责创建具体的产…