Aufgaben

Lösung zu Aufgabe 5.3

Link    Zur fertigen Animation.

Download    Kinematik/Loesungen/verfolgung_kreisbahn.py

"""Simulation und animierte Darstellung eines Verfolgungsproblems.

Der Verfolger (Hund) läuft immer direkt auf den Menschen zu,
der sich auf einer kreisförmigen Bahn bewegt.
"""

import math
import numpy as np
import matplotlib as mpl
import matplotlib.pyplot as plt
import matplotlib.animation

# Startposition (x, y) des Hundes [m].
r0_hund = np.array([28.0, 0.0])
# Geschwindigkeit des Hundes [m/s].
v0_hund = 2
# Radius der Kreisbahn, auf der sich der Mensch bewegt [m].
radius = 5.0
# Bahngeschwindigkeit des Menschen [m/s].
v_mensch = 2.5
# Maximale Simulationsdauer [s].
t_max = 40
# Zeitschrittweite [s].
dt = 0.02

# Mindestabstand, bei dem die Simulation abgebrochen wird.
mindestabstand = v0_hund * dt

# Lege Listen an, um die Simulationsergebnisse zu speichern.
t = [0]
r_hund = [r0_hund]
r_mensch = []
v_hund = []

# Schleife der Simulation
while True:
    # Berechne die aktuelle Position des Menschen.
    r_mensch.append(np.array(
        [radius * math.cos(v_mensch / radius * t[-1]),
         radius * math.sin(v_mensch / radius * t[-1])]))

    # Berechne den Geschwindigkeitsvektor des Hundes.
    r_hund_mensch = r_mensch[-1] - r_hund[-1]
    abstand = np.linalg.norm(r_hund_mensch)
    v = v0_hund * r_hund_mensch / np.linalg.norm(r_hund_mensch)
    v_hund.append(v)

    # Beende die Simulation, wenn der Abstand von Hund und
    # Mensch klein genug ist oder die maximale Simulationszeit
    # überschritten ist.
    if (abstand < mindestabstand) or (t[-1] > t_max):
        break

    # Berechne die neue Position des Hundes und die neue Zeit.
    r_hund.append(r_hund[-1] + dt * v)
    t.append(t[-1] + dt)

# Wandele die Listen in Arrays um. Die Zeilen entsprechen den
# Zeitpunkten und die Spalten entsprechen den Koordinaten.
t = np.array(t)
r_hund = np.array(r_hund)
v_hund = np.array(v_hund)
r_mensch = np.array(r_mensch)

# Erzeuge eine Figure und ein Axes-Objekt.
fig = plt.figure(figsize=(8, 4))
ax = fig.add_subplot(1, 1, 1)
ax.set_xlabel('$x$ [m]')
ax.set_ylabel('$y$ [m]')
ax.set_xlim(-6, 30)
ax.set_ylim(-6, 6)
ax.set_aspect('equal')
ax.grid()

# Erzeuge jeweils einen leeren Linienplot für die Bahnkurve des
# Hundes und des Menschen.
plot_bahn_hund, = ax.plot([], [], color='b')
plot_bahn_mensch, = ax.plot([], [], color='r')

# Erzeuge zwei Punktplots für die aktuelle Postion
# von Hund und Mensch.
plot_hund, = ax.plot([], [], 'o', color='blue')
plot_mensch, = ax.plot([], [], 'o', color='red')

# Erzeuge einen Pfeil für die Geschwindigkeit und füge
# diesen zur Axes hinzu.
style = mpl.patches.ArrowStyle.Simple(head_length=6, head_width=3)
pfeil_v = mpl.patches.FancyArrowPatch((0, 0), (0, 0),
                                      color='red',
                                      arrowstyle=style)
ax.add_patch(pfeil_v)


def update(n):
    """Aktualisiere die Grafik zum n-ten Zeitschritt."""
    # Setze den Start- und Endpunkt des Geschwindigkeitspfeils.
    pfeil_v.set_positions(r_hund[n], r_hund[n] + v_hund[n])

    # Aktualisiere die Positionen von Hund und Mensch
    plot_hund.set_data(r_hund[n].reshape(-1, 1))
    plot_mensch.set_data(r_mensch[n].reshape(-1, 1))

    # Plotte die Bahnkurven bis zum aktuellen Zeitpunkt.
    plot_bahn_hund.set_data(r_hund[:n + 1, 0],
                            r_hund[:n + 1, 1])
    plot_bahn_mensch.set_data(r_mensch[:n + 1, 0],
                              r_mensch[:n + 1, 1])

    return (plot_hund, plot_mensch, pfeil_v,
            plot_bahn_hund, plot_bahn_mensch)


# Erzeuge das Animationsobjekt und starte die Animation.
ani = mpl.animation.FuncAnimation(fig, update, frames=t.size,
                                  interval=30, blit=True)
plt.show()