Kalkulator symboliczny: zadanie “Pudełko Pad Thai”#

W tym pliku rozważymy zadanie z geometrii, którego rozwiązanie może być w całości wykonane z wykorzystaniem możliwości symbolicznych pakietu SymPy. Wszystkie obliczenia będą wykonane w rachunku symbolicznym, co powoduje, że nasze wyniki nie są przybliżeniami i wszelkie uzyskane formuły są udowodnione w sposób algebraiczny.

Zadanie:#

Rozważamy pudełko, którego ściany są pięciokątami foremnymi o długości boku 1. Podstawa pudełka jest kwadratem o boku 1. Zakładając, że ściany pudełka stykają się, oblicz jaka jest odległość pomiędzy górnymi wierzchołkami ścian pudełka leżących naprzeciw siebie.

pad_thai

Wykonujemy import z biblioteki SymPy. W przypadku komunikatu o błędzie importu, należy zainstalować bibliotekę w lokalnej instancji Pythona.

from sympy import *

Biblioteka SymPy posiada wbudowane możliwości rachunków symbolicznych na zmiennych oraz obsługę funkcji matematycznych wraz z relacjami pomiędzy nimi. Konstruując rozwiązanie naszego zadania zapoznamy się z szeregiem standardowych funkcjonalności. Rozszerzone możliwości tej biblioteki są wbudowane w pakiet matematyczny SageMath.

Strategia rozwiązania zadania:#

Aby obliczyć pożądaną odległość spróbujemy “wyobrazić” sobie całą sytuację. Powstanie pudełka polega na obróceniu w przestrzeni 3D czterech pięciokątnych ścian o pewien kąt \(\alpha\). Wybór tego kąta jest jednoznacznie ustalony przez warunek stykania się ścian wzdłuż zewnętrznych krawędzi. Nasza strategia wygląda więc następująco:

  1. Generujemy mechanizm, który wyznacza symbolicznie współrzędne wierzchołków pięciokąta foremnego.

  2. Pozycjonujemy cztery kopie takiego pięciokąta na płaszczyźnie.

  3. Korzystając z transformacji obrotu o zadany kąt względem prostej obracamy wybrane ściany o pewien nieznany kąt \(\alpha\).

  4. Formułujemy warunek stykania ścian (warunek ten jest równaniem).

  5. Wyznaczamy kąt z równania (jako wyrażenie symboliczne).

  6. Obliczamy odległość zadanych punktów korzystając z wbudowanych w SymPy tożsamości trygonometrycznych.

Macierz obrotu o zadany kąt#

Macierze w SymPy są inicjalizowane komendą Matrix. Jako argument funkcja ta pobiera listę składającą z się z list tej samej długości (reprezentujących wiersze).

M1=Matrix([[1,2],[3,4]])
M1
\[\begin{split}\displaystyle \left[\begin{matrix}1 & 2\\3 & 4\end{matrix}\right]\end{split}\]

Na macierzach możemy wykonywać standardowe operacje dodawania i mnożenia, operację obliczania śladu, wyznacznika i wiele innych. Użytkownik może sprawdzić aktualną listę zaimplementowanych funkcji za pomocą nazwy obiektu i kropki oraz przez wciśnięcie tabulatora

M1. #<+tabulator>
  File "/tmp/ipykernel_763/2343214753.py", line 1
    M1. #<+tabulator>
        ^
SyntaxError: invalid syntax

Macierz obrotu wokół punktu \((0,0)\) o zadany kąt \(t\) (przeciwnie do ruchu wskazówek zegara) jest macierzą postaci

def obrot(t):
    m=Matrix([[cos(t),-sin(t)],[sin(t),cos(t)]])
    return m
obrot(pi/2) #obrót o 90 stopni przeciwnie do ruchu wskazówek zegara
\[\begin{split}\displaystyle \left[\begin{matrix}0 & -1\\1 & 0\end{matrix}\right]\end{split}\]

Działanie macierzy obrotu możemy przetestować na wektorach (macierzach o wymiarach \(1\times n\)). Wykorzystamy zmienne symboliczne, aby wyrazić działanie w ogólnym przypadku

x,y=symbols('x,y') #za pomocą komendy symbols możemy inicjalizować listę abstrakcyjnych zmiennych symbolicznych

Na zmiennych symbolicznych \(x,y\) możemy wykonywać operacje podstawiania, elementarne operacje oraz wykorzystywać je jako argumenty w funkcjach, a także podstawiać pod nie wartości.

#dodawanie zmiennych
x+y
\[\displaystyle x + y\]
#mnożenie zmiennych
x*y
\[\displaystyle x y\]
#potęgowanie zmiennych
x**y
\[\displaystyle x^{y}\]

UWAGA!

W standardowym Pythonie symbol ^ oznacza operację dodawania bitowego XOR. Do oznaczenia potęgowania używamy symbolu **

#funkcja z abstrakcyjnym argumentem
cos(x)+sin(y)
\[\displaystyle \sin{\left(y \right)} + \cos{\left(x \right)}\]
#podstawienie argumentu
#w celu podstawienie wykorzystujemy strukturę słownika dict, której kluczami są podstawiane zmienne, a wartościami kluczy
#są wartości podstawiane
sin(x).subs({x:y**2+1})
\[\displaystyle \sin{\left(y^{2} + 1 \right)}\]
#wartości symboliczne można przybliżać numerycznie
cos(pi/7).n()
\[\displaystyle 0.900968867902419\]
#możemy konwertować wartości numeryczne do standardowego typu danych float
float(cos(pi/7))
0.9009688679024191

Ciekawostka: Niektóre stałe matematyczne, np. \(\pi\) są wbudowane w SymPy i traktowane jako pewne zmienne symboliczne, które posiadają szczególne własności, np. wartości \(\cos(\pi/n)\) można podać algebraicznie i numerycznie. To bardzo wygodny sposób operowania podstawowymi stałymi matematyki.

Zbadajmy działanie operacji obrotu o kąt \(t\) na wektorze o zmiennych symbolicznych \((x,y)\)

t,x,y=symbols('t,x,y') #UWAGA: nie musimy ponownie deklarować zmiennych symbolicznych, robimy to tylko dla zachowania
                       #przejrzystości kodu
v=Matrix([[x],[y]])
v
\[\begin{split}\displaystyle \left[\begin{matrix}x\\y\end{matrix}\right]\end{split}\]
obrot_o_t=obrot(t)
obrot_o_t
\[\begin{split}\displaystyle \left[\begin{matrix}\cos{\left(t \right)} & - \sin{\left(t \right)}\\\sin{\left(t \right)} & \cos{\left(t \right)}\end{matrix}\right]\end{split}\]
#obrót wektora (x,y) o kąt t
obrot_o_t*v
\[\begin{split}\displaystyle \left[\begin{matrix}x \cos{\left(t \right)} - y \sin{\left(t \right)}\\x \sin{\left(t \right)} + y \cos{\left(t \right)}\end{matrix}\right]\end{split}\]

Zadanie: jakie są współrzędne wektora \((1,3)\) po obrocie o kąt \(\pi/4\) ?

(obrot_o_t*v).subs({x:1,y:3,t:pi/4})
\[\begin{split}\displaystyle \left[\begin{matrix}- \sqrt{2}\\2 \sqrt{2}\end{matrix}\right]\end{split}\]

Zadanie: Korzystając z wektorów i macierzy obrotów wyznacz współrzędne wierzchołków wielokata foremnego (mającego \(n\) boków) na płaszczyźnie, którego boki mają długość 1

Rozwiązanie metodą “żółwia”:

  1. Wystartuj w punkcie \((0,0)\).

  2. Przesuń się o jednostkę w prawo za pomocą wektora \(k=(1,0)\). Jesteś w punkcie \(P_0=(1,0)\).

  3. Obróć głowę żółwia o kąt \(2\cdot \pi/n\).

  4. Przesuń żółwia o jednostkę “do przodu”. Jesteś w punkcie \(P_1=P_0+\theta(t)(k)\)

  5. Powtarzaj punkty (3)-(5) \(n-1\) razy.

Zwróć listę wierzchołków: \(P_0,\ldots, P_{n-1}\).

Uwaga: Operacja \(\theta(t)\) to obrót wektora o zadany kąt \(t\). W naszym przypadku mnożymy wektor kolumnowy z lewej przez macierz obrotu o kąt \(t\).

def NkatForemny(n):
    p0=Matrix([[0],[0]])
    k=Matrix([[1],[0]])
    ob=obrot(2*pi/n)
    nkat=[]
    w=p0+k
    for i in range(0,n):
        nkat.append(w)
        k=ob*k #obróć żółwia
        w+=k #przesuń o jednostkę w nowym kierunku
    return nkat
pretty_print(NkatForemny(3)) #współrzędne trójkąta foremnego
⎡     ⎡1/2⎤     ⎤
⎢⎡1⎤  ⎢   ⎥  ⎡0⎤⎥
⎢⎢ ⎥, ⎢√3 ⎥, ⎢ ⎥⎥
⎢⎣0⎦  ⎢── ⎥  ⎣0⎦⎥
⎣     ⎣2  ⎦     ⎦

Rysowanie z biblioteką matplotlib#

Do rysowania wykorzystamy uniwersalną bibliotekę matplotlib. Na potrzeby prawidłowego rysowania korzystamy z dodatkowej “magicznej komendy”

%matplotlib inline
import matplotlib.pyplot as plt

Rysowanie wielokątów#

Wielokąt w bibliotece matplotlib możemy narysować korzystając z komendy fill. Do narysowania wielokąta przekazujemy osobno jako argumenty wartości współrzędnych \(x\)-owych i \(y\)-owych. Do rozbicia listy wierzcholków na współrzędne używamy komendy

zip(*lista_wierzcholkow)

Uwaga: aby figury nie nachodziły na siebie przesuwamy macierzowo zbiory wierzchołków o odpowiedni wektor \((3i,0)\) dla każdej \(i\)-tej figury.

fig = plt.figure()
for i in range(0,5):
    wielokatli=(list(tuple(wierzcholek+Matrix([[3*i],[0]])) for wierzcholek in NkatForemny(i+3)));
    wielokatli.append(wielokatli[0]) #dla domknięcia dodajemy ostatni wierzchołek równy pierwszemu
    xli,yli=zip(*wielokatli)
    plt.axis('equal') #proporcjonalne osie
    #axis('off')
    plt.fill(xli,yli)
plt.show() #rysowanie skonfigurowanej ilustracji
_images/fa7f89ae84b45a53056e548f85d545732aa6f1a1dbdf5df17a52dbec86ab00b1.png

Teraz jesteśmy już gotowi do wygenerowania naszego modelowego pięciokąta foremnego. Pamiętamy o dodaniu na końcu listy dodatkowego wierzchołka, który jest równy \(P_0\). Jest to potrzebne dla prawidłowego rysowania wielokątów.

pieciokat0=NkatForemny(5)
pieciokat0.append(pieciokat0[0])
pieciokat=[Matrix([[list(y)[0].simplify()],[list(y)[1].simplify()]]) for y in pieciokat0] #upraszczamy formuły

W kodzie powyżej użyliśmy potężnej funkcjonalności

<wyrazenie>.simplify()

która upraszcza w znany SymPy sposób wyrażenia symboliczne. Do uproszczeń stosuje się znane zależności między funkcjami oraz tożsamości. Efekt działania tej komendy zależy w istotny sposób od wyrażenia i jest zawsze taki sam dla tego samego wyrażenia.

wielokatli=list(tuple(wierzcholek) for wierzcholek in NkatForemny(5));
wielokatli.append(wielokatli[0])
xli,yli=zip(*wielokatli)
fig=plt.figure(figsize=[5,5]) #wielkość figury w kierunku x i y
plt.axis('equal')
plt.axis('off') #wyłączenie wyświetlania osi
plt.fill(xli,yli,color='red')
plt.show()
_images/b935b8443a3931190f38d84bc09d3f84d02fef8be33c3bd348afcaafce2600c4.png

Wielokąty możemy na siebie nakładać i kolorować. Poniższy przykład polega na obracaniu pięciokątów o zadany kąt i ustawianiu ich jeden na drugim, ze zmieniającym się kolorem. Kolor jest zadany jako color=(R,G,B), gdzie parametry \(R\), \(G\), \(B\) są liczbami rzeczywistymi w przedziale \(0\) do \(1\).

#wyznaczanie środka ciężkości
def sumM(li): 
    sum0=li[0]
    for el in li[1:]:
        sum0+=el
    return sum0*Rational(1,len(li))

#obrót wielokąta po przesunięcia środka ciężkości do (0,0) i przesunięcie go z powrotem + zastosowanie skali
def obrotSkalaWielokat(wiel,t,skala): #wiel[:-1]==wiel[0]
    s=sumM(wiel[:-1])
    rot=[obrot(t)*((el-s)*skala)+s for el in wiel[:-1]]
    return rot+[rot[0]]


#inicjalizacja figury
fig=plt.figure(figsize=[5,5]) #wielkość figury w kierunku x i y
plt.axis('equal')
#plt.axis('off') #wyłączenie wyświetlania osi

#trzydzieści iteracji rysowania pięciokąta
iteracje=30
for i in range(0,iteracje):
    wielokatli=obrotSkalaWielokat(pieciokat,2*i*pi/(5*iteracje),Rational(iteracje-i,40*iteracje))
    xli,yli=zip(*wielokatli)
    plt.fill(xli,yli,color=(1-(i/iteracje)**(1/2),(i/iteracje)**2,(iteracje-i)/iteracje))
plt.show()
_images/7f4d74edc3091002111ed6f8e2ee1e670abcebf9bccda586777de5bd0a87f012.png

Wracając do naszego głównego zadania, pozycjonujemy cztery pięciokącie na płaszczyźnie, aby stworzyły makietę ścian naszego pudełka. Ponieważ dno pudełka nie będzie nam potrzebne, więc je ignorujemy.

Pozycjonowanie odbywa się za pomocą obrotów o kąty \(0,90,180\) i \(270\) stopni, wraz z odpowiednimi przesunięciami o wektory na płaszczyźnie. Do szybkiego przetransformowania odpowiednich wektorów stosujemy komendę map wraz z argumentem, którym jest funkcja transformująca i lista (w tym przypadku pieciokat).

pi1=list(map(lambda x: x+Matrix([[0],[1]]),pieciokat))
pi2=list(map(lambda x: obrot(-pi/2)*x+Matrix([[1],[1]]),pieciokat))
pi3=list(map(lambda x: obrot(-pi)*x+Matrix([[1],[0]]),pieciokat))
pi4=list(map(lambda x: obrot(-3/2*pi)*x+Matrix([[0],[0]]),pieciokat))
#definiujemy pomocniczą funkcję rysującą pojedynczy wielokąt
def polygon_draw(li,color):
    wielokatli=list(tuple(wierzcholek) for wierzcholek in li);
    wielokatli.append(wielokatli[0])
    xli,yli=zip(*wielokatli)
    plt.fill(xli,yli,color)

fig=plt.figure(figsize=[5,5])
plt.axis('equal')
polygon_draw(pi1,color='red')
polygon_draw(pi2,color='blue')
polygon_draw(pi3,color='green')
polygon_draw(pi4,color='black')
plt.show()
_images/669c75a2483717e966cf9476dd15098abfafbd24b5b11edc1ec3e150e6026720.png

Teraz czeka nas trudniejsze zadanie. Musimy opracować wzory na obracanie naszych pięciokątów w trzech wymiarach, wokół krawędzi, która styka się z białym kwadratem.

Nasze pięciokąty zyskają dodatkową, trzecią współrzędną, równą 0. Po obrocie ich współrzędne będą zazwyczaj niezerowe.

Obrót wokół osi \(X\) odbywa się w ten sposób, że wektor \((x,y,z)\) zachowuje współrzędną \(y\). Współrzędne \((x,z)\) obracamy za pomocą macierzy obrotu \(\theta(t)\) o zadany kąt \(t\).

x,y,z,t=symbols('x,y,z,t')
v=Matrix([[x],[y],[z]]) #wektor poddany transformacji
vtmp=obrot(t)*Matrix([v[0],v[2]]) #obrót tylko na 0 i 2 współrzędnej
vobr=Matrix([[vtmp[0]],[v[1]],[vtmp[1]]]) #złączenie ponowne w wektor 3D
vobr
\[\begin{split}\displaystyle \left[\begin{matrix}x \cos{\left(t \right)} - z \sin{\left(t \right)}\\y\\x \sin{\left(t \right)} + z \cos{\left(t \right)}\end{matrix}\right]\end{split}\]

Bazując na powyższych kalkulacjach definiujemy zatem funkcję

def ObrotWX0(v,t):
    x,y,z=list(v)
    return Matrix([[x*cos(t) - z*sin(t), y, x*sin(t)+z*cos(t)]])

Podobnie wyznaczamy obrót wokół osi Y

def ObrotWY0(v,t):
    x,y,z=list(v)
    return Matrix([[x, y*cos(t) - z*sin(t), y*sin(t)+z*cos(t)]])

Teraz chcemy obrócić wokół osi zadanej równaniem \(x=1\). W związku z tym, najpierw przesuwamy nasze punktu o \(1\) jednostkę w lewo, obracamy, a następnie wracamy o jedną jednostkę w prawo. Tak uzyskana transformacja będzie obrotem o zadany kąt wokół osi przechodzącej przez \(x=1\).

x,y,z,t=symbols('x,y,z,t')
v=Matrix([[x],[y],[z]]) #wektor poddany transformacji
vtmp=obrot(t)*Matrix([v[0]-1,v[2]]) #obrót tylko na 0 i 2 współrzędnej
vobr=Matrix([[vtmp[0]+1],[v[1]],[vtmp[1]]]) #złączenie ponowne w wektor 3D
vobr
\[\begin{split}\displaystyle \left[\begin{matrix}- z \sin{\left(t \right)} + \left(x - 1\right) \cos{\left(t \right)} + 1\\y\\z \cos{\left(t \right)} + \left(x - 1\right) \sin{\left(t \right)}\end{matrix}\right]\end{split}\]

Przedstawiamy rezultat w formie stosownej funkcji

def ObrotWX1(v,t):
    x,y,z=list(v)
    return Matrix([[-z*sin(t)+(x-1)*cos(t) +1 , y, z*cos(t) + (x-1)*sin(t)]])

Analogiczny rachunek pozwala nam skonstruować funkcję obrotu wokół osi zadanej równaniem \(y=1\)

def ObrotWY1(v,t):
    x,y,z=list(v)
    return Matrix([[x, 1 - cos(t) + y*cos(t)  - z*sin(t), z*cos(t)  - sin(t) + y*sin(t)]])

Teraz mając zadany wektor oraz kąt obrotu, możemy śledzić zmiany konkretnych wektorów pod wpływem obrotów. Przykładowo

t=Symbol('t')
ObrotWX0([1,1,0],t)
\[\displaystyle \left[\begin{matrix}\cos{\left(t \right)} & 1 & \sin{\left(t \right)}\end{matrix}\right]\]
ObrotWY0([1,1,0],t)
\[\displaystyle \left[\begin{matrix}1 & \cos{\left(t \right)} & \sin{\left(t \right)}\end{matrix}\right]\]
ObrotWX1([2,0,0],t)
\[\displaystyle \left[\begin{matrix}\cos{\left(t \right)} + 1 & 0 & \sin{\left(t \right)}\end{matrix}\right]\]
ObrotWY1([0,2,0],t)
\[\displaystyle \left[\begin{matrix}0 & \cos{\left(t \right)} + 1 & \sin{\left(t \right)}\end{matrix}\right]\]

Potrzebujemy teraz biblioteki do rysowania kształtów w 3D

from mpl_toolkits.mplot3d import Axes3D

Wyrysujmy w 3D trajektorie ruchu czterech wierzchołków pod wpływem obrotu o zadany kąt. Funkcja arange pozwala nam wybrać częstotliwość próbkowania liczb z zadanego przedziału.

fig = plt.figure()
ax = fig.gca(projection='3d')
from numpy import arange

x,y,z=list(ObrotWX0([1,1,0],t))
xli=list(map(lambda v:x.subs({t:v}),arange(-pi/2,0,0.1)))
yli=list(map(lambda v:y.subs({t:v}),arange(-pi/2,0,0.1)))
zli=list(map(lambda v:z.subs({t:v}),arange(-pi/2,0,0.1)))
ax.plot(xli, yli, zli)

x,y,z=list(ObrotWY0([1,1,0],t))
xli=list(map(lambda v:x.subs({t:v}),arange(0,pi/2,0.1)))
yli=list(map(lambda v:y.subs({t:v}),arange(0,pi/2,0.1)))
zli=list(map(lambda v:z.subs({t:v}),arange(0,pi/2,0.1)))
ax.plot(xli, yli, zli)

x,y,z=list(ObrotWX1([2,0,0],t))
xli=list(map(lambda v:x.subs({t:v}),arange(0,pi/2,0.1)))
yli=list(map(lambda v:y.subs({t:v}),arange(0,pi/2,0.1)))
zli=list(map(lambda v:z.subs({t:v}),arange(0,pi/2,0.1)))
ax.plot(xli, yli, zli)

x,y,z=list(ObrotWY1([0,2,0],t))
xli=list(map(lambda v:x.subs({t:v}),arange(0,pi/2,0.1)))
yli=list(map(lambda v:y.subs({t:v}),arange(0,pi/2,0.1)))
zli=list(map(lambda v:z.subs({t:v}),arange(0,pi/2,0.1)))
ax.plot(xli, yli, zli)

plt.show()
---------------------------------------------------------------------------
TypeError                                 Traceback (most recent call last)
/tmp/ipykernel_763/4052560376.py in <cell line: 2>()
      1 fig = plt.figure()
----> 2 ax = fig.gca(projection='3d')
      3 from numpy import arange
      4 
      5 x,y,z=list(ObrotWX0([1,1,0],t))

TypeError: gca() got an unexpected keyword argument 'projection'
<Figure size 432x288 with 0 Axes>

Wprowadźmy teraz cztery pięciokąty jako trójwymiarowe figury. Będziemy na nich wykonywać operacje obrotu (na ich wierzchołkach), wokół wybranych wcześniej osi.

pi1_3d=list(map(lambda x: Matrix(list(x)+[0]),pi1))
pi2_3d=list(map(lambda x: Matrix(list(x)+[0]),pi2))
pi3_3d=list(map(lambda x: Matrix(list(x)+[0]),pi3))
pi4_3d=list(map(lambda x: Matrix(list(x)+[0]),pi4))

Funkcjonalność SymPy pozwala nam obrócić dany pięciokąt o abstrakcyjny kąt \(t\).

t=symbols('t')
pol1_3d=list(map(lambda x: ObrotWY1(x,t),pi1_3d)) #obrót przeciwny do ruchu wskazówek zegara
pol2_3d=list(map(lambda x: ObrotWX1(x,t),pi2_3d)) #obrót przeciwny do ruchu wskazówek zegara
pol3_3d=list(map(lambda x: ObrotWY0(x,-t),pi3_3d)) #obrót zgodny z ruchem wskazówek zegara
pol4_3d=list(map(lambda x: ObrotWX0(x,-t),pi4_3d)) #obrót zgodny z ruchem wskazówek zegara

Efekt takiej operacji jest skomplikowanym wyrażeniem wierzchołków jako funkcji zależnych od \(t\).

Uwaga: rezygnujemy z wektorów kolumnowych, od teraz będziemy posługiwać się tylko wektorami wierszowymi

pol1_3d
[Matrix([[1, 1, 0]]),
 Matrix([[sqrt(5)/4 + 3/4, -cos(t) + (sqrt(2*sqrt(5) + 10)/4 + 1)*cos(t) + 1, -sin(t) + (sqrt(2*sqrt(5) + 10)/4 + 1)*sin(t)]]),
 Matrix([[1/2, -cos(t) + (1 + (sqrt(2) + sqrt(10))*sqrt(sqrt(5) + 5)/8)*cos(t) + 1, -sin(t) + (1 + (sqrt(2) + sqrt(10))*sqrt(sqrt(5) + 5)/8)*sin(t)]]),
 Matrix([[1/4 - sqrt(5)/4, -cos(t) + (sqrt(2*sqrt(5) + 10)/4 + 1)*cos(t) + 1, -sin(t) + (sqrt(2*sqrt(5) + 10)/4 + 1)*sin(t)]]),
 Matrix([[0, 1, 0]]),
 Matrix([[1, 1, 0]])]

Wreszcie jesteśmy gotowi rozwiązać nasze zadanie. Mając dane dwa sąsiednie pięciokąty pod wpływem obrotu, czekamy na moment, gdy parametr \(t\) spełnia warunek, że drugi wierzchołek pierwszego pięciokąta pokryje się z czwartym wierzchołkiem drugiego pięciokąta. Wtedy nasze pudełko domyka się.

Chcemy zatem, aby następujące trzy wyrażenia przyjęły wartość \(0\).

#boki zetkną się, gdy wyrażenia poniżej zerują się
wyrazenia=list(pol1_3d[1]-pol2_3d[3])
eq1=Eq(wyrazenia[0],0) #komenda Eq służy do formowania równań
eq2=Eq(wyrazenia[1],0)
eq3=Eq(wyrazenia[2],0)
eq1
\[\displaystyle - \frac{\sqrt{2 \sqrt{5} + 10} \cos{\left(t \right)}}{4} - \frac{1}{4} + \frac{\sqrt{5}}{4} = 0\]
eq2
\[\displaystyle - \cos{\left(t \right)} + \left(\frac{\sqrt{2 \sqrt{5} + 10}}{4} + 1\right) \cos{\left(t \right)} - \frac{\sqrt{5}}{4} + \frac{1}{4} = 0\]
eq3
\[\displaystyle - \sin{\left(t \right)} - \frac{\sqrt{2 \sqrt{5} + 10} \sin{\left(t \right)}}{4} + \left(\frac{\sqrt{2 \sqrt{5} + 10}}{4} + 1\right) \sin{\left(t \right)} = 0\]

Upraszczanie wyrażeń#

Nasze równania \(eq_1,eq_2,eq_3\) możemy uprościć stosując komendę .simplify(). Zauważymy, że dwa pierwsze równania są równoważne. Trzecie upraszcza się do postaci \(0=0\).

eq1.simplify()
\[\displaystyle \frac{\sqrt{2 \sqrt{5} + 10} \cos{\left(t \right)}}{4} - \frac{\sqrt{5}}{4} + \frac{1}{4} = 0\]
eq2.simplify()
\[\displaystyle \frac{\sqrt{2 \sqrt{5} + 10} \cos{\left(t \right)}}{4} - \frac{\sqrt{5}}{4} + \frac{1}{4} = 0\]
(eq1.simplify())==(eq2.simplify())
True
eq3.simplify()
\[\displaystyle \text{True}\]

Pozostaje nam zatem wyznaczyć wartość kąta \(t\) z wykorzystaniem równania \(1\). Do tego wykorzystamy funkcję solve, która równanie symboliczne rozwiązuje ze względu na wskazane zmienne. Zachowanie tej komendy będzie zależało w istotny sposób od wybranego równania.

#rozwiązujemy równanie ze względu na t (rozwiązanie dokładne)
solv=solve(eq1,t)

Obiekt rozwiązania solv przechowuje listę wszystkich rozwiązań. Są dokłądnie 2 - jedno odpowiadające obrotowi naszych ścian w górę, drugie obrotowi w dół. Zauważmy, że rozwiązania są wyrażone symbolicznie za pomocą funkcji arcus cosinus.

len(solv)==2
True
solv[0]
\[\displaystyle - \operatorname{acos}{\left(\frac{- \sqrt{2} + \sqrt{10}}{2 \sqrt{\sqrt{5} + 5}} \right)} + 2 \pi\]
solv[1]
\[\displaystyle \operatorname{acos}{\left(\frac{- \sqrt{2} + \sqrt{10}}{2 \sqrt{\sqrt{5} + 5}} \right)}\]

Gdy interesuje nas wartość numeryczna tych kątów wyrażona w stopniach, możemy posłużyć się przybliżeniem.

(solv[0]*180/pi).n()
\[\displaystyle 288.960709881923\]
(solv[1]*180/pi).n()
\[\displaystyle 71.0392901180775\]

Naszym “magicznym” kątem będzie wybór podyktowany obrotem w górę.

magiczny_kat=solv[1]

Teraz zobaczmy, że faktycznie znaleziony przez nas kąt odpowiada prawidłowemu złożeniu pudełka.

from mpl_toolkits.mplot3d.art3d import Poly3DCollection
def VertFloat(li):
    xl,yl,zl=zip(*li)
    xl=[float(x) for x in xl]
    yl=[float(x) for x in yl]
    zl=[float(x) for x in zl]
    return [list(zip(xl,yl,zl))]
fig = plt.figure()
ax = Axes3D(fig)

#zakres
ax.set_xlim3d(-3,3)
ax.set_ylim3d(-3,3)
ax.set_zlim3d(0,3)

#ustawienia kamery
ax.dist=5
ax.elev=8
ax.azim=170

#wierzchołki wielokątów
tt=magiczny_kat
pol1_3d_float=[VertFloat(list(map(lambda x: ObrotWY1(x,tt),pi1_3d))),['red','black']]
pol2_3d_float=[VertFloat(list(map(lambda x: ObrotWX1(x,tt),pi2_3d))),['blue','black']]
pol3_3d_float=[VertFloat(list(map(lambda x: ObrotWY0(x,-tt),pi3_3d))),['green','black']]
pol4_3d_float=[VertFloat(list(map(lambda x: ObrotWX0(x,-tt),pi4_3d))),['black','white']]
#print(verts)

#dodawanie figur
for verts in [pol1_3d_float,pol2_3d_float,pol3_3d_float,pol4_3d_float]:
    ax.add_collection3d(Poly3DCollection(verts[0],facecolor =verts[1][0],edgecolor=verts[1][1]))
plt.show()
/tmp/ipykernel_763/2991210840.py:17: MatplotlibDeprecationWarning: The dist attribute was deprecated in Matplotlib 3.6 and will be removed two minor releases later.
  ax.dist=5
<Figure size 432x288 with 0 Axes>

Obliczamy odległość wierzchołków#

Pozostaje nam zatem ostatnie zadanie. Obliczymy odległość naprzeciwległych wierzchołków. Wybieramy je jako trzeci wierzchołek pierwszego pięciokąta i trzeci wierzchołek trzeciego pięciokąta.

wierzcholek1=pol1_3d[2]
wierzcholek2=pol3_3d[2]

Odległość wierzchołków od siebie to długość wektora ich różnicy

roznica=wierzcholek1-wierzcholek2

Współrzędne pierwsza i trzecia wektora roznica zerują się. Jedno zerowanie wynika automatycznie, drugie uzyskujemy po zastosowaniu metody simplify()

roznica[0]
\[\displaystyle 0\]
roznica[1].simplify()
\[\displaystyle \frac{\sqrt{2 \sqrt{5} + 10} \cos{\left(t \right)}}{4} + \frac{\sqrt{10 \sqrt{5} + 50} \cos{\left(t \right)}}{4} + 1\]
roznica[2].simplify()
\[\displaystyle 0\]
(wierzcholek1-wierzcholek2)[1].factor()
\[\displaystyle \frac{\sqrt{2} \sqrt{\sqrt{5} + 5} \cos{\left(t \right)} + \sqrt{10} \sqrt{\sqrt{5} + 5} \cos{\left(t \right)} + 4}{4}\]
(wierzcholek1-wierzcholek2)[2].factor()
\[\displaystyle 0\]

Zatem odległość między wierzchołkami wynosi ostatecznie abs(roznica[1]). Obliczmy wprost to wyrażenie dla naszego kąta styku obliczonego wcześniej.

(roznica)[1].subs({t:magiczny_kat}).expand()
\[\displaystyle 2\]

Zatem udowodniliśmy matematycznie, że odległość między wybranymi wierzchołkami wynosi dokładnie 2. Wykorzystana przez nas biblioteka obliczeń symbolicznych SymPy pozwala nam na tę absolutną konluzję.

Inny sposób obliczenia odległosci#

W naszych obliczeniach wyliczenie kąta \(t\) dla styku nie było tak istotne. Kluczowe było, że obliczyliśmy wartość funkcji \(cos(t)\) dla tego kąta. To pozwala nam inaczej podejść do poprzednich rachunków i wyrugować z użycia funkcji solve odwracanie funkcji cosinus.

sol=(roznica).subs({cos(t):solve(eq1,cos(t))[0]})
sol.simplify()
sol
\[\displaystyle \left[\begin{matrix}0 & 2 & 0\end{matrix}\right]\]

Na koniec obejrzyjmy jeszcze raz całą sytuację z wybranego kąta. Korzystamy w naszej funkcji rysującej dodatkowo z interaktywnych widżetów zawartych w bibliotece ipywidgets

from ipywidgets import interactive
from mpl_toolkits.mplot3d.art3d import Line3DCollection

def VertFloat(li):
    xl,yl,zl=zip(*li)
    xl=[float(x) for x in xl]
    yl=[float(x) for x in yl]
    zl=[float(x) for x in zl]
    return [list(zip(xl,yl,zl))]
def f(tt,dist,elev,azim):
    fig = plt.figure(figsize=[3,3])
    ax = Axes3D(fig)

    #zakres
    ax.set_xlim3d(-3,3)
    ax.set_ylim3d(-3,3)
    ax.set_zlim3d(0,3)

    #ustawienia kamery
    ax.dist=dist
    ax.elev=elev
    ax.azim=azim

    #wierzchołki wielokątów

    pol1_3d=[VertFloat(list(map(lambda x: ObrotWY1(x,tt),pi1_3d))),['red','black']]
    pol2_3d=[VertFloat(list(map(lambda x: ObrotWX1(x,tt),pi2_3d))),['blue','black']]
    pol3_3d=[VertFloat(list(map(lambda x: ObrotWY0(x,-tt),pi3_3d))),['green','black']]
    pol4_3d=[VertFloat(list(map(lambda x: ObrotWX0(x,-tt),pi4_3d))),['black','white']]

    #dodawanie figur
    for verts in [pol1_3d,pol2_3d,pol3_3d,pol4_3d]:
        ax.add_collection3d(Poly3DCollection(verts[0],facecolor =verts[1][0],edgecolor=verts[1][1]))
    w1=[float(x) for x in ObrotWY1(pi1_3d[2],tt)]
    w2=[float(x) for x in ObrotWY0(pi3_3d[2],-tt)]  
    ax.add_collection3d(Line3DCollection([[w1,w2]], colors='black',linewidth=3,linestyle='dashed'))
    plt.show()

interactive_plot = interactive(f, tt=(0, float(magiczny_kat),0.01),dist=(0,10,0.1),elev=(0,10,0.1),azim=(0,360,1))
output = interactive_plot.children[-1]
output.layout.height = '350px'
interactive_plot