网站首页 > 技术文章 正文
我们做通讯时,常常要将变量转成字节数组,拷贝进入发送缓冲区(也是字节数组),再进行发送。接收端的变量切分必须准确,否则,收到后也无法正确解析。
由于存在字节对齐的规则,我们查看结构变量的字节大小时,常常发现比所有变量的字节之和要大。这也意味着,如果我们用指针直接访问结构变量时,其成员的位置并不一定是按照成员变量的字节顺序排列,因此,我们需要进行一些确切性的变换来让这些字节按照成员的顺序和大小进行排列。
此处以汇川AM系列PLC的编程软件InoProShop为例(它是CODESYS平台,支持大部分的CODESYS功能和语法),阐述CODESYS里3种获取结构变量里数组的方法。
此处我们定义了一个结构:
TYPE DUT_SEND_DATA_Normal :
STRUCT
STAMP :UDINT;//单位为微秒的时间戳 起始地址:0
data1: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 THEN
sendDataNormal.data1:=sendDataNormal.data1+1;
IF UINT_TO_INT( sendDataNormal.data1) >=30000 THEN
sendDataNormal.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 DO
pTarget^:=pSource^;
pTarget:=pTarget+1;
pSource:=psource+1;
END_FOR
pSource:=ADR(sendDataNormal.data1);
FOR i:=0 TO SIZEOF(sendDataNormal.data1)-1 BY 1 DO
pTarget^:=pSource^;
pTarget:=pTarget+1;
pSource:=psource+1;
END_FOR
pSource:=ADR(sendDataNormal.data2);
FOR i:=0 TO SIZEOF(sendDataNormal.data2)-1 BY 1 DO
pTarget^:=pSource^;
pTarget:=pTarget+1;
pSource:=psource+1;
END_FOR
pSource:=ADR(sendDataNormal.data3);
FOR i:=0 TO SIZEOF(sendDataNormal.data3)-1 BY 1 DO
pTarget^:=pSource^;
pTarget:=pTarget+1;
pSource:=psource+1;
END_FOR
三、利用联合类型(Union)进行操作
联合类型是一种新的数据结构,可以定义同一起始地址的不同变量类型(包括字节数组),操作方法如下:
1、定义几种变量的数据结构:
TYPE union_udint :
UNION
Value:UDINT;
Bytes:ARRAY[0..3] OF BYTE;
END_UNION
END_TYPE
TYPE union_uint :
UNION
Value:UINT;
Bytes:ARRAY[0..1] OF BYTE;
END_UNION
END_TYPE
TYPE union_real :
UNION
Value:REAL;
Bytes:ARRAY[0..3] OF BYTE;
END_UNION
END_TYPE
TYPE union_lreal :
UNION
Value:LREAL;
Bytes:ARRAY[0..7] OF BYTE;
END_UNION
END_TYPE
2、定义新的数据结构
TYPE DUT_SEND_DATA:
STRUCT
STAMP :union_udint;//单位为微秒的时间戳 起始地址:0
data1: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 THEN
sendData.data1.Value:=sendData.data1.Value+1;
IF UINT_TO_INT( sendData.data1.Value) >=30000 THEN
sendData.data1.Value:=0;
END_IF
sendData.data3.Value:=UINT_TO_LREAL(sendData.data1.Value);
END_IF
//数据打包到发送缓冲器,直接操作字节数组。
pArray:=0;
FOR i:=0 TO SIZEOF(sendData.STAMP.Bytes)-1 BY 1 DO
id_SendBuffer[pArray]:=sendData.STAMP.Bytes[i];
pArray:=pArray+1;
END_FOR
FOR i:=0 TO SIZEOF(sendData.data1.Bytes)-1 BY 1 DO
id_SendBuffer[pArray]:=sendData.data1.Bytes[i];
pArray:=pArray+1;
END_FOR
FOR i:=0 TO SIZEOF(sendData.data2.Bytes)-1 BY 1 DO
id_SendBuffer[pArray]:=sendData.data2.Bytes[i];
pArray:=pArray+1;
END_FOR
FOR i:=0 TO SIZEOF(sendData.data3.Bytes)-1 BY 1 DO
id_SendBuffer[pArray]:=sendData.data3.Bytes[i];
pArray:=pArray+1;
END_FOR
2023年10月1日
猜你喜欢
- 2024-09-20 非常详细!如何理解表格存储的多版本、生命周期和有效版本偏差
- 2024-09-20 6种快速统计代码执行时间的方法,真香
- 2024-09-20 Java 开发者最困惑的四件事(java开发遇到问题如何解决)
- 2024-09-20 【Java多线程】定时器Timer(java定时器线程池)
- 2024-09-20 还在用new Date计算任务执行时间?强烈建议使用这个API
- 2024-09-20 “抄”代码,再也不用上谷歌复制粘贴了
- 2024-09-20 java获取当前时间的四种方法代码实例
- 2024-09-20 撸完这篇线程池,我快咳血了(线程池有什么用)
- 2024-09-20 JAVA轮询遍历两个数组进行比较(遍历数组 java)
- 2024-09-20 蒙圈了?System.currentTimeMillis()存在性能问题
- 1514℃桌面软件开发新体验!用 Blazor Hybrid 打造简洁高效的视频处理工具
- 563℃Dify工具使用全场景:dify-sandbox沙盒的原理(源码篇·第2期)
- 508℃MySQL service启动脚本浅析(r12笔记第59天)
- 486℃服务器异常重启,导致mysql启动失败,问题解决过程记录
- 485℃启用MySQL查询缓存(mysql8.0查询缓存)
- 465℃「赵强老师」MySQL的闪回(赵强iso是哪个大学毕业的)
- 445℃mysql服务怎么启动和关闭?(mysql服务怎么启动和关闭)
- 442℃MySQL server PID file could not be found!失败
- 最近发表
- 标签列表
-
- c++中::是什么意思 (83)
- 标签用于 (65)
- 主键只能有一个吗 (66)
- c#console.writeline不显示 (75)
- pythoncase语句 (81)
- es6includes (73)
- windowsscripthost (67)
- apt-getinstall-y (86)
- node_modules怎么生成 (76)
- c++int转char (75)
- static函数和普通函数 (76)
- el-date-picker开始日期早于结束日期 (70)
- js判断是否是json字符串 (67)
- checkout-b (67)
- c语言min函数头文件 (68)
- asynccallback (71)
- localstorage.removeitem (74)
- vector线程安全吗 (70)
- & (66)
- java (73)
- js数组插入 (83)
- mac安装java (72)
- eacces (67)
- 查看mysql是否启动 (70)
- 无效的列索引 (74)