Kollision zweier Kugeln (2D): Unterschied zwischen den Versionen

aus GlossarWiki, der Glossar-Datenbank der Fachhochschule Augsburg
Keine Bearbeitungszusammenfassung
 
(10 dazwischenliegende Versionen desselben Benutzers werden nicht angezeigt)
Zeile 43: Zeile 43:


In Folgenden wird stets der Fall $t=1$ angenommen. Alle Aussagen, die für einen ''beliebigen'' Geschwindigkeitsvektor
In Folgenden wird stets der Fall $t=1$ angenommen. Alle Aussagen, die für einen ''beliebigen'' Geschwindigkeitsvektor
$v$ gelten, gelten dann natürlich auch für $v' := tv$.
$v$ gelten, gelten dann natürlich auch für $v' := sv$, wobei $s$ ein beliebiger skalarer Wert sei.


==[[A-posteriori-Kollisionserkennung]] von zwei Kugeln==
==[[A-posteriori-Kollisionserkennung]] von zwei Kugeln==
Beispiel: [https://glossar.hs-augsburg.de/beispiel/tutorium/2018/ball/WK_Ball02/web/index04c.html WK_Ball02: index40c.html]
<source lang="javascript">
const  EPSILON = Number.EPSILON;


<source lang="javascript">
function collisionCircleCircle(p_c1, p_c2)
  function collisionBallBall(p_b1, p_b2)
{ // l_n: Normalenvektor (Verbindung zwischen den beiden Kugeln)
  {
  let l_nx = p_c2.x - p_c1.x;
    // l_n: Normalenvektor (Verbindung zwischen den beiden Kugelnn)
  let l_ny = p_c2.y - p_c1.y;
    var l_nx = p_b2.x() - p_b1.x();
 
    var l_ny = p_b2.y() - p_b1.y();
  // Abstand der beiden Kugeln
  let l_dist = Math.sqrt(l_nx * l_nx + l_ny * l_ny);
    
    
    // Abstand der beiden Kugeln
  // Die Kugeln dürfen sich nicht an derselben Stelle befinden,
     var l_dist = Math.sqrt( l_nx * l_nx + l_ny * l_ny );
  // weil sie sonst nicht entlang der nicht-existenten Normalen
  // auseinandergezogen werden können.
  // Dieser Fall sollte nur sehr selten eintreten (z.B. wenn eine
  // neue Kugel genau an der Stelle einer existenten Kugel erstellt wird).
  if (l_dist < EPSILON)
  { p_c2.x = p_c2.x + p_c2.r;
    l_nx += p_c2.r;
     l_dist = Math.sqrt(l_nx * l_nx + l_ny * l_ny);
  }
    
    
    // Die Kugeln dürfen sich nicht an derselben Stelle befinden,
  // Kugeln kollidieren, wenn der Abstand kleiner gleich der Summe
    // weil sie sonst nicht entlang der nicht-existenten Normalen
  // der Radien beider Kugeln ist.
    // auseinandergezogen werden können.
  if (l_dist <= p_c1.r + p_c2.r)
    // Dieser Fall sollte nur sehr selten eintreten (z.B. wenn eine
  { // Normalenvektor wird normalisiert: |l_n| = 1
    // neue Kugel genau an der Stelle einer existenten Kugel erstellt wird).
    l_nx /= l_dist;
    if (l_dist < EPSILON)
     l_ny /= l_dist;
    {
      p_b2.x = p_b2.x+ p_b2.r;
      l_nx += p_b2.r;
      l_dist = Math.sqrt( l_nx * l_nx + l_ny * l_ny );
     };
      
      
     // Kugeln kollidieren, wenn der Abstand kleiner gleich der Summe
     // Tangentialvektor (senkrecht zu Normalenvektor, zwischen beiden Kugeln)
    // der Radien beider Kugeln ist.
    let l_tx = l_ny;
    if (l_dist <= p_b1.getR() + p_b2.getR())
    let l_ty = -l_nx;
    {
   
      // Normalenvektor wird normalisiert: |l_n| = 1
    // Summe der Massen beider Kugeln
      l_nx /= l_dist;
    let l_sm1m2 = p_c1.m + p_c2.m;
      l_ny /= l_dist;
   
       
    // Überlappung der beiden Kugeln
      //Tangentialvektor (senkrecht zu Normalenvektor, zwischen beiden Kugeln)
    let l_overlap = (p_c1.r + p_c2.r) - l_dist;
      var l_tx = l_ny;
   
      var l_ty = -l_nx;
    // Verschieben der beiden Kugeln entlang der Normalen,
       
    // so dass sie sich nicht mehr überlappen!
      // Summe der Massen beider Kugeln
    // Die Massen werden berücksichtigt. Schwerere Kreise werden
      var l_sm1m2 = p_b1.m + p_b2.m;
    // weniger weit verschoben.
 
    l_overlap += 2;    /* fight penetration */
      // Überlappung der beiden Kugeln
    l_overlap *= 1.001; /* fight penetration */
      var l_overlap = ( p_b1.r+ p_b2.r ) - l_dist;
    p_c1.x = p_c1.x - l_nx * l_overlap * (p_c2.m / l_sm1m2);
 
    p_c1.y = p_c1.y - l_ny * l_overlap * (p_c2.m / l_sm1m2);
      // Verschieben der beiden Kugeln entlang der Normalen,
    p_c2.x = p_c2.x + l_nx * l_overlap * (p_c1.m / l_sm1m2);
      // so dass sie sich nicht mehr überlappen!
    p_c2.y = p_c2.y + l_ny * l_overlap * (p_c1.m / l_sm1m2);
      p_b1.x = p_b1.x - l_nx * l_overlap * (p_b2.m/l_sm1m2) * (1+EPSILON));
   
      p_b1.y = p_b1.y - l_ny * l_overlap * (p_b2.m/l_sm1m2) * (1+EPSILON));
    // Zerlegung der Geschwindigkeitsvektoren in Normalen- und
      p_b2.x = p_b2.x + l_nx * l_overlap * (p_b1.m/l_sm1m2) * (1+EPSILON));
    // Tangentialanteil: v=sn*n+st*t, wobei
      p_b2.y = p_b2.y + l_ny * l_overlap * (p_b1.m/l_sm1m2) * (1+EPSILON));
    // v Vektor, n Normalenvektor, t Tagentialvektor und
 
    // sn, st zwei skalare Werte sind.
      // Zerlegung der Geschwindigkeitsvektoren in Normalen- und  
    // Es gilt: v*n = sn*(n*n)+st*(t*n) = sn, da t*n=0 und n*n = 1
      // Tangentialanteil: v=sn*n+st*t, wobei  
    // Es gilt: v*t = sn*(n*t)+st*(t*t) = st, da t*n=0 und t*t = 1
      // v Vektor, n Normalenvektor, t Tagentialvektor und
    // Also ist: sn = v*n und st=v*t
      // sn, st zwei skalare Werte sind.
   
      // Es gilt: v*n = sn*(n*n)+st*(t*n) = sn, da t*n=0 und n*n = 1
    // Ball 1: Zerlegung des Geschwindigkeitsvektors in n- und t-Anteil
      // Es gilt: v*t = sn*(n*t)+st*(t*t) = st, da t*n=0 und t*t = 1
    let l_sn1 = l_nx * p_c1.vx + l_ny * p_c1.vy;
      // Also ist: sn = v*n und st=v*t
    let l_st1 = l_tx * p_c1.vx + l_ty * p_c1.vy;
 
   
      // Ball 1: Zerlegung des Geschwindigkeitsvektors in n- und t-Anteil
    let l_n1x = l_nx * l_sn1; // Normalenvektor-Anteil von p_c1.vx
      var l_sn1 = l_nx * p_b1.vx + l_ny * p_b1.vy;
    let l_n1y = l_ny * l_sn1; // Normalenvektor-Anteil von p_c1.vy
      var l_st1 = l_tx * p_b1.vx + l_ty * p_b1.vy;
   
 
    let l_t1x = l_tx * l_st1; // Tangentialvektor-Anteil von p_c1.vx
      var l_n1x = l_nx * l_sn1; // Normalenvektor-Anteil von p_b1.vx
    let l_t1y = l_ty * l_st1; // Tangentialvektor-Anteil von p_c1.vy
      var l_n1y = l_ny * l_sn1;
   
     
    // Ball 2: Zerlegung des Geschwindigkeitsvektors in n- und t-Anteil
      var l_t1x = l_tx * l_st1; // Tangentialvektor-Anteil von p_b1.vx
    let l_sn2 = l_nx * p_c2.vx + l_ny * p_c2.vy;
      var l_t1y = l_ty * l_st1;
    let l_st2 = l_tx * p_c2.vx + l_ty * p_c2.vy;
       
   
      // Ball 2: Zerlegung des Geschwindigkeitsvektors in n- und t-Anteil
    let l_n2x = l_nx * l_sn2; // Normalenvektor-Anteil von p_c2.vx
      var l_sn2 = l_nx * p_b2.vx + l_ny * p_b2.vy;
    let l_n2y = l_ny * l_sn2; // Normalenvektor-Anteil von p_c2.vy
      var l_st2 = l_tx * p_b2.vx + l_ty * p_b2.vy;
   
           
    let l_t2x = l_tx * l_st2; // Tangentialvektor-Anteil von p_c2.vx
      var l_n2x = l_nx * l_sn2; // Normalenvektor-Anteil von p_b2.vx
    let l_t2y = l_ty * l_st2; // Tangentialvektor-Anteil von p_c2.vy
      var l_n2y = l_ny * l_sn2;
   
         
    // Der Impulserhaltungssatz
      var l_t2x = l_tx * l_st2; // Tangentialvektor-Anteil von p_b2.vx
    //  m1*v1 + m2*v2 = m1*v1' + m2*v2'
      var l_t2y = l_ty * l_st2;
    //  (wobei m1, m2 = Massen der Körper
           
    //    und v1, v2, v1', v2' die Geschwindigkeiten)
      // Der Impulserhaltungssatz
    // und der Energieerhaltungssatz
      //  m1*v1 + m2*v2 = m1*v1' + m2*v2'  
    //  0,5*m1*v1² + 0,5*m2*v2² = 0,5*m1*v1'² + 0,5*m2*v2'²
      //  (wobei m1, m2 = Massen der Körper  
    // führen nach einfachen mathematischen Umformungen zu
      //    und v1, v2, v1', v2' die Geschwindigkeiten)
    // folgenden Beziehungen (für den eindimensionalen Fall):
      // und der Energieerhaltsungssatz
    //  v1' = 2*(m1*v1+m2*v2)/(m1+m2) - v1
      //  0,5*m1*v1² + 0,5*m2*v2² = 0,5*m1*v1'² + 0,5*m2*v2'²
    //  v2' = 2*(m1*v1+m2*v2)/(m1+m2) - v2
      // führen nach einfachen mathematischen Umformungen zu
    //  2*(m1*v1+m2*v2)/(m1+m2) ist die Geschwindigkeit des
      // folgenden Beziehungen (für den eindimensionalen Fall):
    //                          gemeinsamen Schwerpunktes
      //  v1' = 2*(m1*v1+m2*v2)/(m1+m2) - v1
    //                          (center of gravity).  
      //  v2' = 2*(m1*v1+m2*v2)/(m1+m2) - v2
    // Im zweidimensionalen Fall gilt, dass die Kollision entlang
      //  2*(m1*v1+m2*v2)/(m1+m2) ist die Geschwindigkeit des  
    // der Normalen erfolgt. Die tangentialen Anteile der der
      //                          gemeinsamen Schwerpunktes.
    // Bewegungsrichtungen werden unverändert übernommen.
      // Im zweidimensionalen Fall gilt, dass die Kollision entlang
   
      // der Normalen erfolgt. Die tangentialen Anteile der der  
    let l_vcgx = 2*(p_c1.m * l_n1x + p_c2.m * l_n2x) / l_sm1m2;
      // Bewegungsrichtungen werden unverändert übernommen.
    let l_vcgy = 2*(p_c1.m * l_n1y + p_c2.m * l_n2y) / l_sm1m2;
       
   
      var l_vspx = 2*(p_b1.m*l_n1x+p_b2.m*l_n2x)/l_sm1m2;
    p_c1.vx = l_vcgx - l_n1x + l_t1x;
      var l_vspy = 2*(p_b1.m*l_n1y+p_b2.m*l_n2y)/l_sm1m2;
    p_c1.vy = l_vcgy - l_n1y + l_t1y;
       
    p_c2.vy = l_vcgx - l_n2x + l_t2x;
      p_b1.vx = l_vspx - l_n1x + l_t1x;
    p_c2.vy = l_vcgy - l_n2y + l_t2y;
      p_b1.vy = l_vspy - l_n1y + l_t1y;
   
      p_b2.vy = l_vspx - l_n2x + l_t2x;
    //// Alternative Berechnung:
      p_b2.vy = l_vspy - l_n2y + l_t2y;
    // Differenz der Massen beide Kugeln
       
    // let l_dm2m1 = p_c2.m - p_c1.m;
      //// Alternative Berechnung:
    // p_c1.vx = (l_n2x*p_c2.m*2-l_n1x*l_dm2m1)/l_sm1m2 + l_t1x;
      // Differenz der Massen beide Kugeln
    // p_c1.vy = (l_n2y*p_c2.m*2-l_n1y*l_dm2m1)/l_sm1m2 + l_t1y;
      // var l_dm2m1 = p_b2.m - p_b1.m;
    // p_c2.vx = (l_n1x*p_c1.m*2+l_n2x*l_dm2m1)/l_sm1m2 + l_t2x;
      // p_b1.vx = (l_n2x*p_b2.m*2-l_n1x*l_dm2m1)/l_sm1m2 + l_t1x;
    // p_c2.vy =(l_n1y*p_c1.m*2+l_n2y*l_dm2m1)/l_sm1m2 + l_t2y;
      // p_b1.vy = (l_n2y*p_b2.m*2-l_n1y*l_dm2m1)/l_sm1m2 + l_t1y;
    return true;
      // p_b2.vx = (l_n1x*p_b1.m*2+l_n2x*l_dm2m1)/l_sm1m2 + l_t2x;
      // p_b2.vy =(l_n1y*p_b1.m*2+l_n2y*l_dm2m1)/l_sm1m2 + l_t2y;
   }
   }
  else
  { return false; }
}
}
</source>
</source>

Aktuelle Version vom 23. November 2018, 19:52 Uhr

Dieser Artikel wird derzeit von einem Autor gründlich bearbeitet. Die Inhalte sind daher evtl. noch inkonsistent.

Dieser Artikel erfüllt die GlossarWiki-Qualitätsanforderungen nur teilweise:

Korrektheit: 1
(nur rudimäntär überprüft)
Umfang: 1
(zu gering)
Quellenangaben: 1
(fehlen großteils)
Quellenarten: 3
(gut)
Konformität: 5
(ausgezeichnet)

Grundannahmen

Es sei ein zweidimensionales Ball-Objekt b gegeben, das folgende Attribute habe:

  • x, y: Position (Einheit: Pixel)
  • vx, vy: Geschwindigkeit in $x$- und $y$-Richtung (Einheit: Pixel/s)
  • r: Radius (Einheit: Pixel)
  • m: Masse (die Einheit ist irrelevant, sie muss nur für alle Bälle dieselbe sein)

Die Frequenz der Neuberechnung der Positionen aller im System vorhandener Bälle sei f, d. h., die Positionen werden f mal pro Sekunde neuberechnet. Wenn es zu keiner Kollision kommt, wird der Ball um vx/f bzw. vy/f Pixel in $x$- bzw. $y$-Richtung verschoben:

  b.x += b.vx/f;
  b.y += b.vy/f;

Je höher die Frequent f ist, desto genauer wird die Bewegungsbahn der Bälle simuliert.

Im Folgenden seinen $s_1 = (s_{1x},s_{1y})$ und $s_2 = (s_{2x},s_{2y})$ die Startpositionen zweier Balle und $v_1 = (v_{1x},v_{1y})$ und $v_2 = (v_{2x},v_{2y})$ die zugehörigen Geschwindigkeitsvektoren.

Es gilt also, dass sich die beiden Bälle zu Zeitpunkt $t_0$ an den Positionen $s_1$ und $s_2$ befinden. Eine Sekunde später, d. h. zum Zeitpunkt $t_0+1$ befinden sich sich an den Positionen $s_1+v_1$ und $s_2+v_2$, sofern sie nicht kollidieren. Wenn die Neuberechnung mit einer Frequenz $f$ Hz erfolgt, d. h., wenn die Periodendauer $t = 1/f$ beträgt, dann befinden sich die Bälle nach der Berechnung an den Positionen $s_1+tv_1$ und $s_2+tv_2$, sofern sie nicht kollidieren.

In Folgenden wird stets der Fall $t=1$ angenommen. Alle Aussagen, die für einen beliebigen Geschwindigkeitsvektor $v$ gelten, gelten dann natürlich auch für $v' := sv$, wobei $s$ ein beliebiger skalarer Wert sei.

A-posteriori-Kollisionserkennung von zwei Kugeln

Beispiel: WK_Ball02: index40c.html

const  EPSILON = Number.EPSILON;

function collisionCircleCircle(p_c1, p_c2)
{ // l_n: Normalenvektor (Verbindung zwischen den beiden Kugeln)
  let l_nx = p_c2.x - p_c1.x;
  let l_ny = p_c2.y - p_c1.y;
  
  // Abstand der beiden Kugeln
  let l_dist = Math.sqrt(l_nx * l_nx + l_ny * l_ny);
  
  // Die Kugeln dürfen sich nicht an derselben Stelle befinden,
  // weil sie sonst nicht entlang der nicht-existenten Normalen
  // auseinandergezogen werden können.
  // Dieser Fall sollte nur sehr selten eintreten (z.B. wenn eine
  // neue Kugel genau an der Stelle einer existenten Kugel erstellt wird).
  if (l_dist < EPSILON)
  { p_c2.x = p_c2.x + p_c2.r;
    l_nx += p_c2.r;
    l_dist = Math.sqrt(l_nx * l_nx + l_ny * l_ny);
  }
  
  // Kugeln kollidieren, wenn der Abstand kleiner gleich der Summe
  // der Radien beider Kugeln ist.
  if (l_dist <= p_c1.r + p_c2.r)
  { // Normalenvektor wird normalisiert: |l_n| = 1
    l_nx /= l_dist;
    l_ny /= l_dist;
    
    // Tangentialvektor (senkrecht zu Normalenvektor, zwischen beiden Kugeln)
    let l_tx =  l_ny;
    let l_ty = -l_nx;
    
    // Summe der Massen beider Kugeln
    let l_sm1m2 = p_c1.m + p_c2.m;
    
    // Überlappung der beiden Kugeln
    let l_overlap = (p_c1.r + p_c2.r) - l_dist;
    
    // Verschieben der beiden Kugeln entlang der Normalen,
    // so dass sie sich nicht mehr überlappen!
    // Die Massen werden berücksichtigt. Schwerere Kreise werden
    // weniger weit verschoben.
    l_overlap += 2;     /* fight penetration */ 
    l_overlap *= 1.001; /* fight penetration */
    p_c1.x = p_c1.x - l_nx * l_overlap * (p_c2.m / l_sm1m2);
    p_c1.y = p_c1.y - l_ny * l_overlap * (p_c2.m / l_sm1m2);
    p_c2.x = p_c2.x + l_nx * l_overlap * (p_c1.m / l_sm1m2);
    p_c2.y = p_c2.y + l_ny * l_overlap * (p_c1.m / l_sm1m2);
    
    // Zerlegung der Geschwindigkeitsvektoren in Normalen- und
    // Tangentialanteil: v=sn*n+st*t, wobei
    // v Vektor, n Normalenvektor, t Tagentialvektor und
    // sn, st zwei skalare Werte sind.
    // Es gilt: v*n = sn*(n*n)+st*(t*n) = sn, da t*n=0 und n*n = 1
    // Es gilt: v*t = sn*(n*t)+st*(t*t) = st, da t*n=0 und t*t = 1
    // Also ist: sn = v*n und st=v*t
    
    // Ball 1: Zerlegung des Geschwindigkeitsvektors in n- und t-Anteil
    let l_sn1 = l_nx * p_c1.vx + l_ny * p_c1.vy;
    let l_st1 = l_tx * p_c1.vx + l_ty * p_c1.vy;
    
    let l_n1x = l_nx * l_sn1; // Normalenvektor-Anteil von p_c1.vx
    let l_n1y = l_ny * l_sn1; // Normalenvektor-Anteil von p_c1.vy
    
    let l_t1x = l_tx * l_st1; // Tangentialvektor-Anteil von p_c1.vx
    let l_t1y = l_ty * l_st1; // Tangentialvektor-Anteil von p_c1.vy
    
    // Ball 2: Zerlegung des Geschwindigkeitsvektors in n- und t-Anteil
    let l_sn2 = l_nx * p_c2.vx + l_ny * p_c2.vy;
    let l_st2 = l_tx * p_c2.vx + l_ty * p_c2.vy;
    
    let l_n2x = l_nx * l_sn2; // Normalenvektor-Anteil von p_c2.vx
    let l_n2y = l_ny * l_sn2; // Normalenvektor-Anteil von p_c2.vy
    
    let l_t2x = l_tx * l_st2; // Tangentialvektor-Anteil von p_c2.vx
    let l_t2y = l_ty * l_st2; // Tangentialvektor-Anteil von p_c2.vy
    
    // Der Impulserhaltungssatz
    //   m1*v1 + m2*v2 = m1*v1' + m2*v2'
    //   (wobei m1, m2 = Massen der Körper
    //    und v1, v2, v1', v2' die Geschwindigkeiten)
    // und der Energieerhaltungssatz
    //   0,5*m1*v1² + 0,5*m2*v2² = 0,5*m1*v1'² + 0,5*m2*v2'²
    // führen nach einfachen mathematischen Umformungen zu
    // folgenden Beziehungen (für den eindimensionalen Fall):
    //   v1' = 2*(m1*v1+m2*v2)/(m1+m2) - v1
    //   v2' = 2*(m1*v1+m2*v2)/(m1+m2) - v2
    //   2*(m1*v1+m2*v2)/(m1+m2) ist die Geschwindigkeit des
    //                           gemeinsamen Schwerpunktes
    //                           (center of gravity).    
    // Im zweidimensionalen Fall gilt, dass die Kollision entlang
    // der Normalen erfolgt. Die tangentialen Anteile der der
    // Bewegungsrichtungen werden unverändert übernommen.
    
    let l_vcgx = 2*(p_c1.m * l_n1x + p_c2.m * l_n2x) / l_sm1m2;
    let l_vcgy = 2*(p_c1.m * l_n1y + p_c2.m * l_n2y) / l_sm1m2;
    
    p_c1.vx = l_vcgx - l_n1x + l_t1x;
    p_c1.vy = l_vcgy - l_n1y + l_t1y;
    p_c2.vy = l_vcgx - l_n2x + l_t2x;
    p_c2.vy = l_vcgy - l_n2y + l_t2y;
    
    //// Alternative Berechnung:
    // Differenz der Massen beide Kugeln
    // let l_dm2m1 = p_c2.m - p_c1.m;
    // p_c1.vx = (l_n2x*p_c2.m*2-l_n1x*l_dm2m1)/l_sm1m2 + l_t1x;
    // p_c1.vy = (l_n2y*p_c2.m*2-l_n1y*l_dm2m1)/l_sm1m2 + l_t1y;
    // p_c2.vx = (l_n1x*p_c1.m*2+l_n2x*l_dm2m1)/l_sm1m2 + l_t2x;
    // p_c2.vy =(l_n1y*p_c1.m*2+l_n2y*l_dm2m1)/l_sm1m2 + l_t2y;
    return true;
  }
  else
  { return false; }
}

A-priori-Kollisionserkennung von zwei Kugeln

Kollision zweier Kugeln
Der Tunnel-Effekt

Quellen

  1. Kowarschick (WebProg): Wolfgang Kowarschick; Vorlesung „Web-Programmierung“; Hochschule: Hochschule Augsburg; Adresse: Augsburg; Web-Link; 2024; Quellengüte: 3 (Vorlesung)