Möglichkeit cartesianAngle(Sin[c],-Cos[c]) weiter zu optimieren?



  • Wenn ich das richtig sehe ist das eben die atan2 Funktion. Nur selbst nachgebaut.

    Wenn aus cartesischen Koordinaten der Winkel am Ursprung zwischen der Linie vom Ursprung zum durch von (x,y) gegebenen Punkt berechnet werden soll, nimmt man dafür üblicherweise den atan(y/x). Der atan hat aber das Problem, den Winkel eben nicht in jedem Quartal des Koordinatenkreuzes richtig darstellen zu können. Daher nimmt man hier den atan2(y, x) (oder man macht eine ewig lange Fallunterscheidung um das Ergebnis zu korrigieren).

    Beachte, dass in den C und C++ Implementationen erst der y-Parameter und dann der x-Parameter übergeben wird.



  • hab mal ein bisschen mit C++ und Mathematica gespielt

    hab noch einen Fehler in meiner cartesian_angle-Portierung fuer Mathematica gefunden

    so sieht sie jetzt aus

    cartesianAngle[x_, y_] := 
    Module[{},
    If[ x > 0 && y > 0, Atan[ y / x ],
      If[ x < 0 && y > 0, Atan[ -x / y ] + Pi/2, 
        If[ x < 0 && y < 0, Atan[ y / x ] + Pi,
          If[ x > 0 && y < 0, Atan[ -x / y ] + 3*Pi/2, 
            If[ x == 0 && y > 0, Pi/2, 
              If[ x <  0 && y == 0, Pi,
                If[ x == 0 && y < 0, 3*Pi/2,
                  0
                ]
              ]
            ]
          ]
        ]
      ]
    ]
    ]
    

    {π2x=0y>0πx<0y=03π2x=0y<0Atan(xy)+π2x<0y>0Atan(xy)+3π2x>0y<0Atan(yx)x>0y>0Atan(yx)+πx<0y<0\begin{array}{cc} \{ & \begin{array}{cc} \frac{\pi }{2} & x=0\land y>0 \\ \pi & x<0\land y=0 \\ \frac{3 \pi }{2} & x=0\land y<0 \\ \text{Atan}\left(-\frac{x}{y}\right)+\frac{\pi }{2} & x<0\land y>0 \\ \text{Atan}\left(-\frac{x}{y}\right)+\frac{3 \pi }{2} & x>0\land y<0 \\ \text{Atan}\left(\frac{y}{x}\right) & x>0\land y>0 \\ \text{Atan}\left(\frac{y}{x}\right)+\pi & x<0\land y<0 \\ \end{array} \\ \end{array}

    hab mal in C++ die std::atan2 probiert:

    ( x, y)
    ( 1, 1) c++-cartesian_angle: 0.785398 std::atan2: 0.785398 equal: 1
    (-1, 1) c++-cartesian_angle: 2.356194 std::atan2: 2.356194 equal: 1
    (-1,-1) c++-cartesian_angle: 3.926991 std::atan2: -2.356194 equal: 0 !!!
    ( 1,-1) c++-cartesian_angle: 5.497787 std::atan2: -0.785398 equal: 0 !!!
    ( 0, 1) c++-cartesian_angle: 1.570796 std::atan2: 1.570796 equal: 1
    (-1, 0) c++-cartesian_angle: 3.141593 std::atan2: 3.141593 equal: 1
    ( 0,-1) c++-cartesian_angle: 4.712389 std::atan2: -1.570796 equal: 0 !!!
    ( 0, 0) c++-cartesian_angle: 0.000000 std::atan2: 0.000000 equal: 1
    

    Geprüft:
    1. die Mathematica-Funktions-Ausgabe mit den selben Punkten lierfert die gleichen Ergebnisse wie c++-cartesian_angle
    2. die std::atan2 liefert bei manchen Punkten andere Ergebnisse als die c++-cartesian_angle
    3. Mathematica ArcTan liefert für alle Punkte die gleiche Ergebnisse wie std::atan2

    die c++ cartesian_angle ist wohl doch keine 1:1 entsprechung zu atan2 - oder mache ich was anderes falsch?



  • Die Frage ist ob die C++-cartesian_angle vielleicht auch nicht richtig oder "äquivalent" implementierung zu atan2 ist

    wenn ich atan2 Einsetzen könnte würde sich die specialTpDirectionCartensianAngleSincos auf {Cos[c], Sin[c]} reduzieren - was sehr schön wäre 🙂



  • Das sieht aus wie der entsprechende Gegenwinkel unter anderem Vorzeichen.

    Die atan2 Funktion nimmt bei negativen y, den Winkel mit negativem Vorzeichen. Hast also einen Wertebereich von [-pi, pi]

    Bei deiner cartesian_angle Implementierung nimmst du jeweils den positiven Winkel und hast den Wertebereich von [0, 2pi]

    Bei folgendem Link findest du 'ne ganz nette Grafik, die den Wertebereich des atan2 wiedergibt: https://de.mathworks.com/help/fixedpoint/ref/atan2.html


  • Mod

    Und bevor jetzt groß um den Unterschied herum gearbeitet wird: Wenn der Code zur Weiterverarbeitung des Ergebnisses etwas taugt, sollte es ihm egal sein, ob man 350 Grad oder -10 Grad sagt. Ist schließlich die gleiche Richtung, die man damit beschreibt.



  • Ich hab jetzt mal Testweise die cartesian_angle durch std::atan2 ersetzt und die Berechnung liefern immer noch identische Ergebnisse zum Original 🙂 - auch die Reduzierung auf nur Cos[c],Sin[c] funktioniert - weniger geht nicht - noch mal 10-20sek schneller 🙂

    Jetzt muss ich nur noch herausbekommen ob das anderer Verhalten von atan2 an den bestimmten Stellen problematisch sein kann

    bisher vielen Danke für alle Tips


  • Mod

    Lass mich raten: Der Winkel wird dann sowieso wieder in irgendwelche anderen trigonometrischen Funktionen eingesetzt. Falls ja, kann man sich das ganze Theater auch sparen und gewinnt ungeheuer viel Zeit.

    Faustregel: Eine konkrete Winkelangabe ist etwas, das man praktisch niemals innerhalb einer Rechnung benötigt. Einen Winkel tatsächlich auszurechnen ist etwas, das man fast immer nur ganz am Ende macht, damit sich beispielsweise ein Mensch ein schönes Bildchen mit den Ergebnissen zeichnen möchte. Beim Numbercrunching selbst ist es (gewaltige!) Zeitverschwendung, Winkel zu berechnen.



  • Lass mich raten: Der Winkel wird dann sowieso wieder in irgendwelche anderen trigonometrischen Funktionen eingesetzt. Falls ja, kann man sich das ganze Theater auch sparen und gewinnt ungeheuer viel Zeit.

    genau so ist es - es gehen als Parameter {sin(c),-cos(c)} in eine If-Orgie mit Atan,Cot und Tan usw. ein - und nach der Vereinfachung(durch atan2) kommt raus das das Ergebnis einfach immer {cos(c),sin(c)} ist - das hatte ich gehofft

    jetzt ist nur noch eine (wieder sehr aehnliche) Rotations-Party übrig welche dieses {cos(c),sin(c)} nutzt - mal schauen ob das auch so zusammenfällt wie bei diesen Punkt hier

    nochmals Danke



  • Nach meinem Gefühlt reduziert sich dadurch auch die Ungenauigkeiten - und wenn am Ende nur noch sin(c) und cos(c) übrig bleiben gibt es dafür bestimmt auch noch eine schöne SSE2/3/4... sincos, whatever Lösung - danach kann ich immer noch die Eingangsdaten mit n Threads durchkauen - aber eben erst mal schauen das Single-Threaded alles rausgeholt ist


  • Mod

    Wie gesagt, die Lösung ist bestimmt eher beim Algorithmus für das Gesamtproblem bzw. der Mathematik zu suchen, als in der technischen Umsetzung dieses kleinen Abschnitts.



  • Wie gesagt, die Lösung ist bestimmt eher beim Algorithmus für das Gesamtproblem bzw. der Mathematik zu suchen, als in der technischen Umsetzung dieses kleinen Abschnitts.

    Das ist mit klar - ich probiere das auch nur so weil der Algorithmus auch schon mit dieser Vorgehensweise ins fast nichts verdampft - und er auch nicht so gross ist



  • Mal aus Neugierde: Du hast geschrieben, dass es sich um eine 5 Achsen Kinematic handelt.

    Meinst du damit tatsächlich, ein Glied, welches durch 5 Achsen angesteuert wird? Oder eine kinematische Kette mit 5 Gelenken?

    Und geht es nur um die Vorwärts Kinematik, also die Bestimmung des Endpunktes, oder auch die Rückwärtskinematik?

    Ich frage, weil ich mich damit selbst mal rumschlagen musste und mir damals zwischenzeitlich ganz schön einen abgebrochen habe 😃



  • Meinst du damit tatsächlich, ein Glied, welches durch 5 Achsen angesteuert wird? Oder eine kinematische Kette mit 5 Gelenken?

    es sind 2 Glieder mit 3 und 2 Achsen

    Und geht es nur um die Vorwärts Kinematik, also die Bestimmung des Endpunktes, oder auch die Rückwärtskinematik?

    nur Endpunkbestimmung


Anmelden zum Antworten