gpt4 book ai didi

loops - Arduino循环中的奇怪行为

转载 作者:行者123 更新时间:2023-12-01 12:06:46 25 4
gpt4 key购买 nike

我正在 Arduino(Mega2560 板、Arduino 1.8.9、JSN-SR04T-2.0 超声波传感器)中创建距离测量系统。由于传感器不是很稳定,我想对 15 次测量或每 500 毫秒的测量值进行平均,以先发生者为准。这些值存储在一个 15 整数长数组中,并且有一个计数器告诉进行了多少次测量。问题是当完成 15 次测量并且我想取平均值(不包括在代码中,只有 if(...) 部分)时,计数器从 14 跳到一个非常高的值,没有任何解释并从那里继续计数.此外,如果我设置非常高的循环时间(例如 1800 毫秒),有时它会回到 0 而不会进入 if 结构(cntr=0 行所在的位置)。

我将代码精简到最少,以找出可能导致问题的原因。似乎当我使用存储在“valueArrayUS”中的值时,它不起作用。如果我将其注释掉,它就会起作用。我尝试了 2 种不同的方式来使用该值,但都搞砸了。

最小可运行代码是这样的:

int trigPin = 8;    // Trigger
int echoPin = 9; // Echo
int cntr=0; //how many measurements
long startTime=0; //start time of cycle
int valueArrayUS[15]; //max 15 values per cycle
int valueArrayUScp[15]; //copy of valueArrayUS
int sumUS=0;
long duration, cm;
int i=0, j=0;

void setup() {
Serial.begin (9600);
pinMode(trigPin, OUTPUT);
pinMode(echoPin, INPUT);

for (i=0; i<15; i++) {
valueArrayUS[i]=0; //default state
valueArrayUScp[i]=valueArrayUS[i];
}
startTime=millis(); //time counter start
}

void loop() {

//ultrasonic measurement
digitalWrite(trigPin, LOW);
delayMicroseconds(50);
digitalWrite(trigPin, HIGH);
delayMicroseconds(20);
digitalWrite(trigPin, LOW);

duration = pulseIn(echoPin, HIGH, 38000); //if no object is detected -> 38 ms long pulse from sensor -> 38 ms timeout
cm = (duration/2) / 29.1; // Divide by 29.1 or multiply by 0.0343
valueArrayUS[cntr]=cm;

Serial.print("cntr: ");
Serial.print(cntr);
Serial.println();

if ((millis()-startTime>500) || (cntr==15)) { //cycle is ended or buffer full

Serial.print("enters the cycle, cycle time: ");
Serial.println();
Serial.print(millis()-startTime);
Serial.print(" ms ");
Serial.println();
Serial.print("cntr at cycle start: ");
Serial.print(cntr);
Serial.println();

//averaging would be here
for (j=0; j<15; j++) {
valueArrayUScp[j]=valueArrayUS[j]; //THIS IS WHERE IT GOES WRONG
//sumUS=sumUS+valueArrayUS[j]; //THIS IS WHERE IT GOES WRONG
}

for (i=0; i<15; i++) {
valueArrayUS[i]=0; //default state
}
cntr=0;
startTime=millis();
}
else {
cntr=cntr+1;
Serial.print("cntr++ ");
Serial.println();
}
}

您可以看到注释行和它上面的行,它们都破坏了代码。如果两者都被注释掉,它会按预期工作。如果其中任何一个存在,结果如下:

15:51:33.848 -> cntr:  11
15:51:33.894 -> cntr++
15:51:33.894 -> cntr: 12
15:51:33.894 -> cntr++
15:51:33.894 -> cntr: 13
15:51:33.942 -> cntr++
15:51:33.942 -> cntr: 14
15:51:33.942 -> cntr++
15:51:33.942 -> cntr: 117
15:51:33.989 -> cntr++
15:51:33.989 -> cntr: 118
15:51:33.989 -> cntr++
15:51:34.036 -> cntr: 119
15:51:34.036 -> cntr++
15:51:34.036 -> cntr: 120
15:51:34.036 -> cntr++
15:51:34.083 -> cntr: 121
15:51:34.083 -> cntr++
15:51:34.083 -> cntr: 122
15:51:34.083 -> cntr++
15:51:34.129 -> cntr: 123
15:51:34.129 -> cntr++
15:51:34.129 -> cntr: 124
15:51:34.129 -> cntr++
15:51:34.176 -> cntr: 125
15:51:34.176 -> cntr++
15:51:34.176 -> cntr: 126
15:51:34.223 -> enters the cycle, cycle time:
15:51:34.223 -> 557 ms
15:51:34.270 -> cntr at cycle start: 126
15:51:34.270 -> cntr: 0
15:51:34.270 -> cntr++
15:51:34.318 -> cntr: 1

应该是:

15:57:33.132 -> cntr:  14
15:57:33.132 -> cntr++
15:57:33.132 -> cntr: 15
15:57:33.132 -> enters the cycle, cycle time:
15:57:33.179 -> 392 ms
15:57:33.179 -> cntr at cycle start: 15
15:57:33.226 -> cntr: 0
15:57:33.226 -> cntr++
15:57:33.226 -> cntr: 1

是什么原因造成的?

最佳答案

您的循环执行此操作:

{
valueArrayUS[cntr]=cm;
//...
if ((millis()-startTime>500) || (cntr==15)) { //cycle is ended or buffer full
//...
else {
cntr=cntr+1;
}
}

删除其余代码后,更容易看出您:

  1. 使用 cntr
  2. 测试 cntr 是否为 15(即超过数组末尾)
  3. 更新 cntr

通过 loop() 的前 15 次这不是问题,但在第 15 次之后 cntr 在函数结束时变为 15,并在下一次通过时通过你最终这样做的功能:

valueArrayUS[15]=cm;

这意味着你正在写超过数组的末尾,它看起来像内存中 valueArrayUS 之后的下一个东西必须是 cntr,因为你显然是覆盖该值。

这是问题的核心,但一旦发生这种情况,您就会陷入痛苦的世界,因为现在 cntr 的值可能 > 15,因此您的 cntr==15 条件不为真,现在 loop 将在 valueArrayUS 末尾写入传感器值。这将有效地破坏您的所有内存,而在给定运行中究竟覆盖了哪些内存将取决于传感器恰好正在读取的内容。

要修复,您需要重新安排步骤:

  1. 检查cntr是否在边界内
  2. 如果不是,则进行计算并重置 cntr
  3. 如果 cntr 在边界内,那么您可以读取传感器、存储值并更新 cntr

此外,防御性编码:

  • 将您的条件从 cntr == 15 更改为 cntr > 14cntr >= 15
  • 在本地声明其他循环变量,ij,例如:for (int i...

关于loops - Arduino循环中的奇怪行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55747672/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com