arduino - 转换交集面积以生成坐标

标签 arduino processing computational-geometry coordinate-systems trilateration

我正在开展一个项目,我从电阻式触摸屏获取模拟值并将其转换为交叉点。

这是一个示例:enter image description here

这是我使用 Arduino Uno 进行数据收集的代码,以及使用名为“处理”的工具构建点的代码。

#define side1 2
#define side2 3
#define side3 4
#define side4 5
#define contact A0

void setup() {
  pinMode(contact, INPUT); 
  pinMode(side1, OUTPUT);  
  pinMode(side2, OUTPUT);  
  pinMode(side3, OUTPUT);  
  pinMode(side4, OUTPUT);  
  Serial.begin(9600);
}

void loop() {
  int sensorValue1;
  int sensorValue2;
  int sensorValue3;
  int sensorValue4;

  // SENSOR VALUE 1:
  digitalWrite(side1, LOW);
  digitalWrite(side2, HIGH);
  digitalWrite(side3, HIGH);
  digitalWrite(side4, HIGH);
  delay(5);
  for (int i = 0; i < 10; i++){
    sensorValue1 = analogRead(contact);
  }


  // SENSOR VALUE 2:
  digitalWrite(side2, LOW);
  digitalWrite(side3, HIGH);
  digitalWrite(side4, HIGH);
  digitalWrite(side1, HIGH);
  delay(5);
  for (int i = 0; i < 10; i++){
    sensorValue2 = analogRead(contact);
  }


  // SENSOR VALUE 3:
  digitalWrite(side3, LOW);
  digitalWrite(side2, HIGH);
  digitalWrite(side4, HIGH);
  digitalWrite(side1, HIGH);
  delay(5);
  for (int i = 0; i < 10; i++){
    sensorValue3 = analogRead(contact);
  }


  // SENSOR VALUE 2:
  digitalWrite(side4, LOW);
  digitalWrite(side3, HIGH);
  digitalWrite(side2, HIGH);  
  digitalWrite(side1, HIGH);
  delay(5);
  for (int i = 0; i < 10; i++){
    sensorValue4 = analogRead(contact);
  }

  Serial.print(sensorValue1);
  Serial.print(",");
  Serial.print(sensorValue2);
  Serial.print(",");
  Serial.print(sensorValue3);
  Serial.print(",");
  Serial.print(sensorValue4);
  Serial.println();
}

这是构建图表的处理代码。

import processing.serial.*;


Serial myPort;  // The serial port
int maxNumberOfSensors = 4;   
float[] sensorValues = new float[maxNumberOfSensors];
float sensorValueX;
float sensorValueX1;
float sensorValueY;
float sensorValueY1;
int scaleValue = 2;

void setup () { 
  size(600, 600);  // set up the window to whatever size you want
  //println(Serial.list());  // List all the available serial ports
  String portName = "COM5";
  myPort = new Serial(this, portName, 9600);
  myPort.clear();
  myPort.bufferUntil('\n');  // don't generate a serialEvent() until you get a newline (\n) byte
  background(255);    // set inital background
  smooth();  // turn on antialiasing
}


void draw () {
  //background(255);
  //noFill();
  fill(100,100,100,100);
  ellipse(height,0, scaleValue*sensorValues[0], scaleValue*sensorValues[0]);

  ellipse(0,width, scaleValue*sensorValues[1], scaleValue*sensorValues[1]);
  ellipse(height,width, scaleValue*sensorValues[2], scaleValue*sensorValues[2]);
  ellipse(0,0, scaleValue*sensorValues[3], scaleValue*sensorValues[3]);
  //ellipse(sensorValueY, sensorValueX, 10,10);
  //println(sensorValueY,sensorValueX);
  sensorValueX = ((sensorValues[3]*sensorValues[3])-(sensorValues[2]*sensorValues[2])+600*600)/2000;
  sensorValueX1 = ((sensorValues[0]*sensorValues[0])-(sensorValues[1]*sensorValues[1])+600*600)/2000;
sensorValueY = ((sensorValues[3]*sensorValues[3])-(sensorValues[2]*sensorValues[2])+(600*600))/2000;
  sensorValueY1 = ((sensorValues[1]*sensorValues[1])-(sensorValues[0]*sensorValues[0])+(600*600))/2000;

  line(0, scaleValue*sensorValueX, height,scaleValue* sensorValueX);
  line(scaleValue*sensorValueY, 0, scaleValue*sensorValueY, width);
  ellipse(scaleValue*sensorValueY, scaleValue*sensorValueX, 20,20);
  line(0, scaleValue*sensorValueX1, height,scaleValue* sensorValueX1);
  line(scaleValue*sensorValueY1, 0, scaleValue*sensorValueY1, width);
  ellipse(scaleValue*sensorValueY1, scaleValue*sensorValueX1, 20,20);
  println(scaleValue*sensorValueX,scaleValue*sensorValueY);
}


void serialEvent (Serial myPort) {
  String inString = myPort.readStringUntil('\n');  // get the ASCII string

  if (inString != null) {  // if it's not empty
    inString = trim(inString);  // trim off any whitespace
    int incomingValues[] = int(split(inString, ","));  // convert to an array of ints

    if (incomingValues.length <= maxNumberOfSensors && incomingValues.length > 0) {
      for (int i = 0; i < incomingValues.length; i++) {
        // map the incoming values (0 to  1023) to an appropriate gray-scale range (0-255):

        sensorValues[i] = map(incomingValues[i], 0, 1023, 0, width);
        //println(incomingValues[i]+ " " + sensorValues[i]);
      }
    }
  }
}

我想知道如何将这些点的交集转换为坐标?示例:在我向您展示的图像中,我将尺寸参数设置为 (600,600)。是否可以将交集更改为坐标值?目前,我的代码正在打印坐标,但它们是对角线,例如 x 和 y 值相等。我希望 x 和 y 的坐标具有不同的数量,以便我可以获得正方形中不同边的坐标。有人可以帮忙吗?

最佳答案

通过阅读您的代码,我假设您知道所有 n 个传感器的位置以及每个 n 个传感器到目标的距离。所以你本质上想做的是 trilateration (如尼科·谢特勒所述)。换句话说,根据n个点之间的距离确定相对位置。

只是一个快速定义注释,以防混淆:

三边测量至少需要 3 个点和距离。

  • 1 个传感器可提供目标距传感器的距离
  • 2 个传感器为您提供目标的 2 个可能位置
  • 3 个传感器告诉您目标位于 2 个位置中的哪一个

可能想到的第一个解决方案是计算交集 3 个传感器之间将它们视为圆圈。鉴于距离可能存在一些误差,这意味着圆可能并不总是相交。这就排除了这个解决方案。

以下代码已全部在Processing中完成。

我冒昧地做了一个class Sensor .

class Sensor {
    public PVector p; // position
    public float d; // distance from sensor to target (radius of the circle)

    public Sensor(float x, float y) {
        this.p = new PVector(x, y);
        this.d = 0;
    }
}

现在要计算并近似传感器/圆之间的交点,请执行以下操作:

PVector trilateration(Sensor s1, Sensor s2, Sensor s3) { 
    PVector s = PVector.sub(s2.p, s1.p).div(PVector.sub(s2.p, s1.p).mag());
    float a = s.dot(PVector.sub(s3.p, s1.p));

    PVector t = PVector.sub(s3.p, s1.p).sub(PVector.mult(s, a)).div(PVector.sub(s3.p, s1.p).sub(PVector.mult(s, a)).mag());
    float b = t.dot(PVector.sub(s3.p, s1.p));
    float c = PVector.sub(s2.p, s1.p).mag();

    float x = (sq(s1.d) - sq(s2.d) + sq(c)) / (c * 2);
    float y = ((sq(s1.d) - sq(s3.d) + sq(a) + sq(b)) / (b * 2)) - ((a / b) * x);

    s.mult(x);
    t.mult(y);

    return PVector.add(s1.p, s).add(t);
}

哪里s1 , s2 , s3是 3 个传感器中的任何一个,执行以下操作来计算给定传感器之间的交点:

PVector target = trilateration(s1, s2, s3);

虽然可以计算任意数量的传感器之间的交集。您想要包含的传感器越多,它就会变得越来越复杂。尤其是因为你是自己做的。如果您能够使用外部 Java 库,那么事情就会容易得多。

如果您能够使用外部 Java 库,那么我强烈建议使用 com.lemmingapex.trilateration 。然后您就可以通过执行以下操作来计算 4 个传感器之间的交点:

考虑s1 , s2 , s3 , s4作为前面提到的 class Sensor 的实例.

double[][] positions = new double[][] { { s1.x, s1.y }, { s2.x, s2.y }, { s3.x, s3.y }, { s4.x, s4.y } };
double[] distances = new double[] { s1.d, s2.d, s3.d, s4.d };

NonLinearLeastSquaresSolver solver = new NonLinearLeastSquaresSolver(
            new TrilaterationFunction(positions, distances),
            new LevenbergMarquardtOptimizer());
Optimum optimum = solver.solve();

double[] target = optimum.getPoint().toArray();
double x = target[0];
double y = target[1];

以下示例是 trilateration() 的示例我编写的方法,而不是上面库的示例。

示例 1 - 无传感器错误

三个大圆圈是任意 3 个传感器,单个红色圆圈是近似点。

Example 1

示例 2 - 传感器错误

三个大圆圈是任意 3 个传感器,单个红色圆圈是近似点。

Example 2

关于arduino - 转换交集面积以生成坐标,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42824656/

相关文章:

java - "Security settings have blocked a self-signed application from running"作为开发者

连接几何线的算法

arduino 在上次上传后突然显示 "avrdude: ser_open(): can' t 打开设备 "\\.\COM3"

C 到 Pascal 类型转换

java - 在 Java 应用程序中使用处理

python-3.x - noLoop 是否停止执行绘制?

python - 将 3D 椭球拟合到 3D 空间中的点 - 不同的方法,不同的答案

javascript - 沿四边形的周边对点进行排序

无法在赋值中将 'float*' 转换为 'float'

python - 使用串口将 float 从arduino发送到python