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


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

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

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

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

    1. TYPE DUT_SEND_DATA_Normal :
    2. STRUCT
    3. STAMP :UDINT;//单位为微秒的时间戳 起始地址:0
    4. data1:UINT;//UInt类型的数值
    5. data2:REAL;//浮点数类型的数值
    6. data3:LREAL;;//双精度类型的数值
    7. END_STRUCT
    8. END_TYPE

    一、利用M区域进行访问

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

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

    二、利用指针进行操作

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

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

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

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

    1. GetSystemTime(uliTimeUs=>clockus);//获取系统时间(微秒为单位)
    2. sendDataNormal.STAMP:=ULINT_TO_UDINT(clockus);//截取低4字节的值。
    3. sendPulse:=NOT(sendPulse);//发送脉冲
    4. //周期计数
    5. IF sendPulse THEN
    6. sendDataNormal.data1:=sendDataNormal.data1+1;
    7. IF UINT_TO_INT( sendDataNormal.data1) >=30000 THEN
    8. sendDataNormal.data1:=0;
    9. END_IF
    10. END_IF
    11. //双精度格式的周期计数
    12. sendDataNormal.data3:=UINT_TO_LREAL(sendDataNormal.data1);
    13. //数据打包到发送缓冲器,直接操作字节数组。
    14. pTarget:=ADR(id_SendBuffer);
    15. pSource:=ADR(sendDataNormal.STAMP);
    16. FOR i:=0 TO SIZEOF(sendDataNormal.STAMP)-1 BY 1 DO
    17. pTarget^:=pSource^;
    18. pTarget:=pTarget+1;
    19. pSource:=psource+1;
    20. END_FOR
    21. pSource:=ADR(sendDataNormal.data1);
    22. FOR i:=0 TO SIZEOF(sendDataNormal.data1)-1 BY 1 DO
    23. pTarget^:=pSource^;
    24. pTarget:=pTarget+1;
    25. pSource:=psource+1;
    26. END_FOR
    27. pSource:=ADR(sendDataNormal.data2);
    28. FOR i:=0 TO SIZEOF(sendDataNormal.data2)-1 BY 1 DO
    29. pTarget^:=pSource^;
    30. pTarget:=pTarget+1;
    31. pSource:=psource+1;
    32. END_FOR
    33. pSource:=ADR(sendDataNormal.data3);
    34. FOR i:=0 TO SIZEOF(sendDataNormal.data3)-1 BY 1 DO
    35. pTarget^:=pSource^;
    36. pTarget:=pTarget+1;
    37. pSource:=psource+1;
    38. END_FOR

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

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

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

    1. TYPE union_udint :
    2. UNION
    3. Value:UDINT;
    4. Bytes:ARRAY[0..3] OF BYTE;
    5. END_UNION
    6. END_TYPE
    7. TYPE union_uint :
    8. UNION
    9. Value:UINT;
    10. Bytes:ARRAY[0..1] OF BYTE;
    11. END_UNION
    12. END_TYPE
    13. TYPE union_real :
    14. UNION
    15. Value:REAL;
    16. Bytes:ARRAY[0..3] OF BYTE;
    17. END_UNION
    18. END_TYPE
    19. TYPE union_lreal :
    20. UNION
    21. Value:LREAL;
    22. Bytes:ARRAY[0..7] OF BYTE;
    23. END_UNION
    24. END_TYPE

    2、定义新的数据结构

    1. TYPE DUT_SEND_DATA:
    2. STRUCT
    3. STAMP :union_udint;//单位为微秒的时间戳 起始地址:0
    4. data1:union_uint;//UInt类型的数值
    5. data2:union_real;//浮点数类型的数值
    6. data3:union_lreal;//双精度类型的数值
    7. END_STRUCT
    8. END_TYPE

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

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

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

    1. GetSystemTime(uliTimeUs=>clockus);//获取系统时间(微秒为单位)
    2. sendPulse:=NOT(sendPulse);//发送脉冲
    3. sendData.STAMP.Value:=ULINT_TO_UDINT(clockus);//截取低4字节的值。
    4. //周期计数
    5. IF sendPulse THEN
    6. sendData.data1.Value:=sendData.data1.Value+1;
    7. IF UINT_TO_INT( sendData.data1.Value) >=30000 THEN
    8. sendData.data1.Value:=0;
    9. END_IF
    10. sendData.data3.Value:=UINT_TO_LREAL(sendData.data1.Value);
    11. END_IF
    12. //数据打包到发送缓冲器,直接操作字节数组。
    13. pArray:=0;
    14. FOR i:=0 TO SIZEOF(sendData.STAMP.Bytes)-1 BY 1 DO
    15. id_SendBuffer[pArray]:=sendData.STAMP.Bytes[i];
    16. pArray:=pArray+1;
    17. END_FOR
    18. FOR i:=0 TO SIZEOF(sendData.data1.Bytes)-1 BY 1 DO
    19. id_SendBuffer[pArray]:=sendData.data1.Bytes[i];
    20. pArray:=pArray+1;
    21. END_FOR
    22. FOR i:=0 TO SIZEOF(sendData.data2.Bytes)-1 BY 1 DO
    23. id_SendBuffer[pArray]:=sendData.data2.Bytes[i];
    24. pArray:=pArray+1;
    25. END_FOR
    26. FOR i:=0 TO SIZEOF(sendData.data3.Bytes)-1 BY 1 DO
    27. id_SendBuffer[pArray]:=sendData.data3.Bytes[i];
    28. pArray:=pArray+1;
    29. END_FOR

    四、小结

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

    2023年10月5日

  • 相关阅读:
    Redis入门完整教程:Java客户端Jedis
    深入浅出排序算法之计数排序
    (六)centos7案例实战——sonarQube安装及springboot项目集成sonarQube完成代码质量检查
    vector容器中push_back()和emplace_back()函数的区别
    【算法集训 | 暑期刷题营】7.27题---并查集
    用高并发技巧解决redis热key问题
    2024.1.1 IntelliJ IDEA 使用记录
    青岛大学数据结构与算法——第3章
    Django--ORM 常用字段及属性介绍
    vue-qr插件使用 报错You may need an appropriate loader to handle this file type
  • 原文地址:https://blog.csdn.net/chengjl8/article/details/133564593