以汇川中型PLC(AM系列)为例,CODESYS平台变量与字节数组互转的多种方法

本文主要是介绍以汇川中型PLC(AM系列)为例,CODESYS平台变量与字节数组互转的多种方法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

我们做通讯时,常常要将变量转成字节数组,拷贝进入发送缓冲区(也是字节数组),再进行发送。接收端的变量切分必须准确,然后再正确地转换成对应类型的变量,否则,收到后也无法正确解析。

由于存在字节对齐的规则,我们查看结构变量的字节大小时,常常发现比所有变量的字节之和要大。这也意味着,如果我们用指针直接访问结构变量时,其成员的位置并不一定是按照成员变量的字节顺序排列,因此,我们需要进行一些确切性的变换来让这些字节按照成员的顺序和大小进行排列。

本文以汇川AM系列PLC的编程软件InoProShop为例(它是CODESYS平台,支持大部分的CODESYS功能和语法),阐述CODESYS里3种获取结构变量里数组的方法(也可以用于反向转换)。

此处我们定义了一个结构:

TYPE DUT_SEND_DATA_Normal :
STRUCTSTAMP :UDINT;//单位为微秒的时间戳 起始地址:0data1:UINT;//UInt类型的数值data2:REAL;//浮点数类型的数值data3:LREAL;;//双精度类型的数值
END_STRUCT
END_TYPE

一、利用M区域进行访问

M区域有字节编号,可以直接使用。该方法较复杂,但是,适用于V2和V3版本。

在全局变量里定义一个该结构的变量,下载运行,就可以在线看到每个变量的起始地址,然后就可以在程序里将这些字节逐个移入发送缓冲区:

二、利用指针进行操作

每个变量都有内存起始地址,通过指针进行获取,然后进行指针操作,也可以获取变量的字节数组。该方法适用于V2和V3版本,并且可以不需要借助M区域。

1、在主程序里新建局部变量:

clockus:ULINT;
sendPulse:BOOL;
sendDataNormal:DUT_SEND_DATA_Normal;
pSource:POINTER TO BYTE;
pTarget:POINTER TO BYTE;
id_SendBuffer:ARRAY[0..199] OF BYTE;//发送缓冲器。

2、在主程序里增加以下语句:

GetSystemTime(uliTimeUs=>clockus);//获取系统时间(微秒为单位)
sendDataNormal.STAMP:=ULINT_TO_UDINT(clockus);//截取低4字节的值。
sendPulse:=NOT(sendPulse);//发送脉冲//周期计数
IF sendPulse THENsendDataNormal.data1:=sendDataNormal.data1+1;IF UINT_TO_INT( sendDataNormal.data1) >=30000 THENsendDataNormal.data1:=0;END_IF
END_IF//双精度格式的周期计数
sendDataNormal.data3:=UINT_TO_LREAL(sendDataNormal.data1);//数据打包到发送缓冲器,直接操作字节数组。
pTarget:=ADR(id_SendBuffer);pSource:=ADR(sendDataNormal.STAMP);
FOR i:=0 TO SIZEOF(sendDataNormal.STAMP)-1 BY 1 DOpTarget^:=pSource^;pTarget:=pTarget+1;pSource:=psource+1;
END_FORpSource:=ADR(sendDataNormal.data1);
FOR i:=0 TO SIZEOF(sendDataNormal.data1)-1 BY 1 DOpTarget^:=pSource^;pTarget:=pTarget+1;pSource:=psource+1;
END_FORpSource:=ADR(sendDataNormal.data2);
FOR i:=0 TO SIZEOF(sendDataNormal.data2)-1 BY 1 DOpTarget^:=pSource^;pTarget:=pTarget+1;pSource:=psource+1;
END_FORpSource:=ADR(sendDataNormal.data3);
FOR i:=0 TO SIZEOF(sendDataNormal.data3)-1 BY 1 DOpTarget^:=pSource^;pTarget:=pTarget+1;pSource:=psource+1;
END_FOR

三、利用联合类型(Union)进行操作

联合类型是一种新的数据结构,可以定义同一起始地址的不同变量类型(包括字节数组),操作方法如下:

1、定义几种变量类型的数据结构:

TYPE union_udint :
UNIONValue:UDINT;Bytes:ARRAY[0..3] OF BYTE;
END_UNION
END_TYPETYPE union_uint :
UNIONValue:UINT;Bytes:ARRAY[0..1] OF BYTE;
END_UNION
END_TYPETYPE union_real :
UNIONValue:REAL;Bytes:ARRAY[0..3] OF BYTE;
END_UNION
END_TYPETYPE union_lreal :
UNIONValue:LREAL;Bytes:ARRAY[0..7] OF BYTE;
END_UNION
END_TYPE

2、定义新的数据结构

TYPE DUT_SEND_DATA:
STRUCTSTAMP :union_udint;//单位为微秒的时间戳 起始地址:0data1:union_uint;//UInt类型的数值data2:union_real;//浮点数类型的数值data3:union_lreal;//双精度类型的数值
END_STRUCT
END_TYPE

3、在主程序里新建局部变量:

clockus:ULINT;
sendPulse:BOOL;
sendData:DUT_SEND_DATA;
id_SendBuffer:ARRAY[0..199] OF BYTE;//发送缓冲器。
pArray:UINT;
i:UINT;

4、在主程序里增加以下语句:

GetSystemTime(uliTimeUs=>clockus);//获取系统时间(微秒为单位)
sendPulse:=NOT(sendPulse);//发送脉冲
sendData.STAMP.Value:=ULINT_TO_UDINT(clockus);//截取低4字节的值。//周期计数
IF sendPulse THENsendData.data1.Value:=sendData.data1.Value+1;IF UINT_TO_INT( sendData.data1.Value) >=30000 THENsendData.data1.Value:=0;END_IFsendData.data3.Value:=UINT_TO_LREAL(sendData.data1.Value);
END_IF//数据打包到发送缓冲器,直接操作字节数组。
pArray:=0;
FOR i:=0 TO SIZEOF(sendData.STAMP.Bytes)-1 BY 1 DOid_SendBuffer[pArray]:=sendData.STAMP.Bytes[i];pArray:=pArray+1;
END_FORFOR i:=0 TO SIZEOF(sendData.data1.Bytes)-1 BY 1 DOid_SendBuffer[pArray]:=sendData.data1.Bytes[i];pArray:=pArray+1;
END_FORFOR i:=0 TO SIZEOF(sendData.data2.Bytes)-1 BY 1 DOid_SendBuffer[pArray]:=sendData.data2.Bytes[i];pArray:=pArray+1;
END_FORFOR i:=0 TO SIZEOF(sendData.data3.Bytes)-1 BY 1 DOid_SendBuffer[pArray]:=sendData.data3.Bytes[i];pArray:=pArray+1;
END_FOR

四、小结

通过联合类型可以安全、直观地进行变量与字节数组的互相转换,虽然其本质也是指针,但是,这种方式不易犯错误,因此,值得大家掌握。

2023年10月5日

这篇关于以汇川中型PLC(AM系列)为例,CODESYS平台变量与字节数组互转的多种方法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Python判断for循环最后一次的6种方法

《Python判断for循环最后一次的6种方法》在Python中,通常我们不会直接判断for循环是否正在执行最后一次迭代,因为Python的for循环是基于可迭代对象的,它不知道也不关心迭代的内部状态... 目录1.使用enuhttp://www.chinasem.cnmerate()和len()来判断for

Java循环创建对象内存溢出的解决方法

《Java循环创建对象内存溢出的解决方法》在Java中,如果在循环中不当地创建大量对象而不及时释放内存,很容易导致内存溢出(OutOfMemoryError),所以本文给大家介绍了Java循环创建对象... 目录问题1. 解决方案2. 示例代码2.1 原始版本(可能导致内存溢出)2.2 修改后的版本问题在

浅析Rust多线程中如何安全的使用变量

《浅析Rust多线程中如何安全的使用变量》这篇文章主要为大家详细介绍了Rust如何在线程的闭包中安全的使用变量,包括共享变量和修改变量,文中的示例代码讲解详细,有需要的小伙伴可以参考下... 目录1. 向线程传递变量2. 多线程共享变量引用3. 多线程中修改变量4. 总结在Rust语言中,一个既引人入胜又可

四种Flutter子页面向父组件传递数据的方法介绍

《四种Flutter子页面向父组件传递数据的方法介绍》在Flutter中,如果父组件需要调用子组件的方法,可以通过常用的四种方式实现,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录方法 1:使用 GlobalKey 和 State 调用子组件方法方法 2:通过回调函数(Callb

一文详解Python中数据清洗与处理的常用方法

《一文详解Python中数据清洗与处理的常用方法》在数据处理与分析过程中,缺失值、重复值、异常值等问题是常见的挑战,本文总结了多种数据清洗与处理方法,文中的示例代码简洁易懂,有需要的小伙伴可以参考下... 目录缺失值处理重复值处理异常值处理数据类型转换文本清洗数据分组统计数据分箱数据标准化在数据处理与分析过

Java中Object类的常用方法小结

《Java中Object类的常用方法小结》JavaObject类是所有类的父类,位于java.lang包中,本文为大家整理了一些Object类的常用方法,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. public boolean equals(Object obj)2. public int ha

golang1.23版本之前 Timer Reset方法无法正确使用

《golang1.23版本之前TimerReset方法无法正确使用》在Go1.23之前,使用`time.Reset`函数时需要先调用`Stop`并明确从timer的channel中抽取出东西,以避... 目录golang1.23 之前 Reset ​到底有什么问题golang1.23 之前到底应该如何正确的

Redis多种内存淘汰策略及配置技巧分享

《Redis多种内存淘汰策略及配置技巧分享》本文介绍了Redis内存满时的淘汰机制,包括内存淘汰机制的概念,Redis提供的8种淘汰策略(如noeviction、volatile-lru等)及其适用场... 目录前言一、什么是 Redis 的内存淘汰机制?二、Redis 内存淘汰策略1. pythonnoe

Vue项目中Element UI组件未注册的问题原因及解决方法

《Vue项目中ElementUI组件未注册的问题原因及解决方法》在Vue项目中使用ElementUI组件库时,开发者可能会遇到一些常见问题,例如组件未正确注册导致的警告或错误,本文将详细探讨这些问题... 目录引言一、问题背景1.1 错误信息分析1.2 问题原因二、解决方法2.1 全局引入 Element

Python调用另一个py文件并传递参数常见的方法及其应用场景

《Python调用另一个py文件并传递参数常见的方法及其应用场景》:本文主要介绍在Python中调用另一个py文件并传递参数的几种常见方法,包括使用import语句、exec函数、subproce... 目录前言1. 使用import语句1.1 基本用法1.2 导入特定函数1.3 处理文件路径2. 使用ex