arrays - 为什么我只在写入变量时得到 "VBScript runtime error: Overflow"?

标签 arrays import vbscript tia-portal

我需要任何 VBScript 向导的帮助。

我目前正在研究导入/导出功能,这些功能能够在工业 PLC 的本地 SD 卡和 HMI(控制面板)的本地 USB 之间移动数据。这一切都在 Tia-Portal V15.1 编程环境中进行编程。

启动导出功能时,PLC 将其本地 SD 卡中的所有 64135 字节数据分解到传输缓冲区(即多个字节数组(array[array[bytes]])) 在数据 block (DB) 中。传输缓冲区由 HMI 引用,并且可以由内部调用的 VBScript 毫无问题地访问。这是通过引用变量 USBSD_Interface_TransferBuffer.Datapack_# 完成的。使用 FOR 循环,VBScript 对每个元素进行计数,并可以将它们写入连接的 USB 上的日志文件:

'#############################################################################################
'#                            HANDSHAKE TO WRITE DATA TO FILE                                #
'#-------------------------------------------------------------------------------------------#
Dim LOCAL_BUFFERSIZE : LOCAL_BUFFERSIZE = 1599          ' Array[0..1599] = 1600 entries
Const MAX_ARRAYSIZE = 64135                             ' Max number of elements in array
Const MAX_ARRAYCOUNTER = 40                             ' 40 * 1600 = 64000 values
Dim BUFFER_COUNTER : BUFFER_COUNTER = 0
Dim ARRAY_COUNTER : ARRAY_COUNTER = 0
'#############################################################################################

For ARRAY_COUNTER = 0 To MAX_ARRAYCOUNTER
    If ARRAY_COUNTER = MAX_ARRAYCOUNTER Then
        ' Catch case of final datapack only holding 135 elements
        LOCAL_BUFFERSIZE = 135  
    End If
    
    For BUFFER_COUNTER = 0 To LOCAL_BUFFERSIZE
        FILE.WriteLine SmartTags("USBSD_Interface_TransferBuffer.Datapack_" & ARRAY_COUNTER)(BUFFER_COUNTER)
    Next
Next

虽然从数组中读取数据并将数据写入外部文件没有问题,但 Import 函数却不是这样。在这里,我试图做完全相同的事情,但相反。我首先加载文件并将每个条目放入一个变量 (FILEDATA),该变量本质上是一个值数组。然后迭代这些值,并将其写入传输缓冲区中的相应位置:

'#############################################################################################
'#                         HANDSHAKE TO WRITE DATA TO TEMP-DB                                #
'#-------------------------------------------------------------------------------------------#
Const MAX_ARRAYSIZE = 40
Const MAX_ELEMENTS = 64135
Dim LOCAL_BUFFERSIZE : LOCAL_BUFFERSIZE = 1599
Dim ARRAY_COUNTER : ARRAY_COUNTER = 0
Dim BUFFER_COUNTER : BUFFER_COUNTER = 0
Dim ELEMENT_COUNTER : ELEMENT_COUNTER = 0
'#############################################################################################

For ARRAY_COUNTER = 0 To MAX_ARRAYSIZE
    If ARRAY_COUNTER = MAX_ARRAYSIZE Then
        LOCAL_BUFFERSIZE = 135
    End If
    
    For BUFFER_COUNTER = 0 To LOCAL_BUFFERSIZE
        SmartTags("USBSD_Interface_TransferBuffer.Datapack_" & ARRAY_COUNTER)(BUFFER_COUNTER) = CByte(FILEDATA(ELEMENT_COUNTER))
        ELEMENT_COUNTER = ELEMENT_COUNTER + 1
    Next
    
    If ELEMENT_COUNTER >= MAX_ELEMENTS Then
        Exit For
    End If
Next

我目前在运行我的脚本时遇到以下错误:

0x800a0006 - Microsoft VBScript runtime error: Overflow: 'BUFFER_COUNTER'

这只发生在导入阶段,当计数器达到第一个数据包的BUFFER_COUNTER = 99时(即ARRAY_COUNTER = 0) .

出于绝对的绝望,我尝试手动强制执行前 100 个变量,以查看 FOR 循环是否通过以下方式导致问题:

SmartTags("USBSD_Interface_TransferBuffer.Datapack_0")(0) = CByte(FILEDATA(0))
SmartTags("USBSD_Interface_TransferBuffer.Datapack_0")(1) = CByte(FILEDATA(1))
SmartTags("USBSD_Interface_TransferBuffer.Datapack_0")(2) = CByte(FILEDATA(2))
...
SmartTags("USBSD_Interface_TransferBuffer.Datapack_0")(99) = CByte(FILEDATA(99))
SmartTags("USBSD_Interface_TransferBuffer.Datapack_0")(100) = CByte(FILEDATA(100))

... 在位置 99 处导致相同的溢出错误。我知道所有传输缓冲区数组都已实例化,因为我可以单独读/写它们的值。例如,调用:

SmartTags("USBSD_Interface_TransferBuffer.Datapack_0")(99) = CByte(FILEDATA(99))
SmartTags("USBSD_Interface_TransferBuffer.Datapack_0")(100) = CByte(FILEDATA(100))
SmartTags("USBSD_Interface_TransferBuffer.Datapack_0")(101) = CByte(FILEDATA(101))
SmartTags("USBSD_Interface_TransferBuffer.Datapack_0")(102) = CByte(FILEDATA(102))

...没有错误。只有在 VBScript 执行期间我尝试写入数组中超过 98 的元素时,才会出现此错误。

我真的对这个不知所措。如果有人以前遇到过类似问题,我们将不胜感激任何提示/技巧/解决方法!

最佳答案

与其他专业人士就此事交谈后,已找到导致此错误的原因。 “溢出”不是计数器变量的溢出,而是写入太多变量时硬件处理时间/功率的过载情况,时间太短(无所谓Microsoft 错误说明了什么)。

推荐的解决方案是创建一个暂停或时间循环来强制 HMI 停止并处理已经调用的变量,然后再尝试写入新变量。该循环位于主 FOR 循环中,看起来像这样:

If ELEMENT_COUNTER Mod 50 = 0 Then
  Dim WAIT_TIME : WAIT_TIME = Now + 1 / 24 / 3600 ' wait for one second
  Do
  Loop While Now <= WAIT_TIME
End If

不幸的是,这个建议来得太晚了:我已经创建了一个我自己的临时解决方案,无意中绕过了这个问题,没有意识到问题到底是什么。我在 PLC 中创建了一个 50 字节的辅助缓冲区,由 HMI 填充。一旦填满,HMI 就会向 PLC 发送信号,PLC 会系统地用辅助缓冲区中的值填充主缓冲区,然后再向 HMI 发送信号以继续该过程。这种握手一直持续到主缓冲区被填满。

我相信我会让我的临时解决方案更“永久”,不管它比推荐的解决方案慢一点。如果我没记错的话,避免使用暂停和等待函数是编程的一般经验法则,除非调用函数本身是时间相关的。即便如此,也应谨慎对待它们。

关于arrays - 为什么我只在写入变量时得到 "VBScript runtime error: Overflow"?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/72488040/

相关文章:

c++ - 数组下标的无效类型 'int[int]'

python - 如何在同一个包中导入 __init__.py 中定义的类?

vbscript - 通过 VBScript 检测 Windows 驱动器

c# - 如何在不按 Ctrl + C 的情况下将任何应用程序中的选定文本获取到 VB.NET

java - 为什么文件是只读的?

javascript - 将一个数组推到另一个数组的末尾

javascript - 如何补偿仍在进行中的数组迭代中已删除的元素?

c++ - QImage数组的动态分配

html - 如何将svg从文件导入 Angular 为5的组件中?

c++ - 如何从 C 文件调用 C++ 构造函数