Pola figur: całkowanie analityczne i metodą Monte Carlo#

W jaki sposób można obliczyć pole figury o zadanym kształcie, który jest trudny do analitycznego opisu? W tym pliku przedyskutujemy sposób obliczania pól figur oparty na metodach losowych z wykorzystaniem prostych spacerów losowych. Porównamy wyniki z obliczeniami analitycznymi na konkretnym przykładzie.

#ignorowanie komunikatów technicznych
import warnings
warnings.filterwarnings("ignore")

Problem#

Oblicz z dokładnością \(0.01\) pole wewnątrz następującej figury.

x,y=var('x,y')
show(implicit_plot((x**2+y**2)**2-2*(x**2-y**2),(x,-2,2),(y,-2,2),figsize=[5,5],ymax=1,ymin=-1,xmin=-2,xmax=2))
_images/4ee826718dbcd4c740c0ff9eb86161d3f3d28a71ffaec65f8672ddf95a6a77f4.png

Obliczenia analityczne#

W naszym pierwszym podejściu obliczymy pole wewnątrz “ósemki” w sposób dokładny (analityczny). W tym celu musimy rozwiazać uwikłane równanie opisujące krzywą.

Rozwiązanie równania#

Najpierw “rozwiązujemy” nasze równanie uwikłane.

\[(x^2+y^2)^2=2(x^2-y^2)\]
solv=solve((x**2+y**2)**2-2*(x**2-y**2)==0,(x,y))
pretty_print(solv)

Powyższy obiekt przechowuje wszystke “lokalne” rozwiązania naszego równania. Wykorzystamy je do napisania równań funkcji, które kawałkami opisują kształt naszej “ósemki” (precyzyjnie: lemniskaty Bernoulliego).

UWAGA:

Dwie spośród podanych wyżej funkcji rozwiązań nie opisują rzeczywistych części wykresu - ignorujemy je.

y1=solv[2][1].rhs() #wybieramy prawą stronę trzeciego równania
r=y1.variables()[0] #odnosimy się do jego wewnętrznej zmiennej symbolicznej
f=y1.subs({r:x}) #i podstawiamy ustaloną przez nas od początku zmienną x

#sprawdzamy równość podstawienia
assert bool(f==sqrt(-x^2 + sqrt(4*x^2 + 1) - 1))

#to będzie nasza "rozwiązana" funkcja
funkcja=sqrt(-x^2 + sqrt(4*x^2 + 1) - 1)

W jakim prostokącie mieści się wykres lemniskaty?#

Obliczmy teraz formalnie liczby \(a\) i \(b\) takie, że zbiór punktów

\[L=\{(x,y)\in\mathbb{R}^2: (x^2+y^2)^2=2(x^2-y^2)\}\]

spełnia

\[\max_{(x,y)\in L} x = - \min_{(x,y)\in L} x = a\]

oraz

\[\max_{(x,y)\in L} y = - \min_{(x,y)\in L} y = b\]

Wartość parametru \(a\) przy nam się przy obliczaniu pola (całki) metodą analityczną. Parametr \(b\) będzie nam potrzebny również przy obliczaniu pola numerycznie metodą Monte Carlo.

Liczbę \(a\) wyznaczamy określając dziedzinę funkcji \(f(x) = \sqrt{\sqrt{4x^2+1}-x^2-1}\)

#UWAGA: w SageMath można używać zarówno ** jak i ^ jako symbolu potęgowania (w zwykłym Python symbol ^ oznacza
#operację XOR; w SageMath ta operacja jest oznaczona jako ^^)
eq1=0==-x^2 + sqrt(4*x^2 + 1) - 1 #formułujemy zerowanie argumenty pod zewnętrznym pierwiastkiem

#wykonujemy przekształcenia na równaniu eq1, aby doprowadzić je do wygodnej wielomianowej postaci
eq1=eq1.add_to_both_sides(x^2+1)
eq1=eq1*eq1 #podniesienie obu stron do kwadratu
eq1=eq1.add_to_both_sides(-(4*x^2+1))
eq1
(x^2 + 1)^2 - 4*x^2 - 1 == 0
solve(eq1,x)
[x == -sqrt(2), x == sqrt(2), x == 0]

Zatem nasz parametr \(a\) wynosi \(\sqrt{2}\).

Liczbę \(b\) obliczymy wyznaczając maksimum osiągane przez funkcję \(f(x)\) na przedziale \((0,\sqrt{2})\).

Obliczamy pochodną funkcji \(f\) i zauważamy, że mianownik jest dodatni. Do określenia przebiegu funkcji użyjemy informacji o znaku licznika.

pochodnaf=funkcja.derivative(x).factor()
licznik=pochodnaf.numerator()
mianownik=pochodnaf.denominator()
#licznik
pretty_print(licznik)
#mianownik
pretty_print(mianownik)
#wykres licznika pochodnej
show(plot(licznik,(x,0,sqrt(2)),figsize=[3,3]))
_images/6bcdf24a3603f0c2f2d771094c5ec6f6df6f30e8a5abf8a7d1f393fe21a392ae.png

Z postaci funkcji \(-x(\sqrt{4x^2+1}-2)\) wynika, że jest dodatnia dla \(x<c\) i ujemna dla \(x>c\), gdzie \(c\) jest jedynym miejscem zerowym tej funkcji. Obliczmy zatem \(c\).

with assuming(x>0):
    print(solve(licznik==0,x))
[
x == 1/2*sqrt(3)
]

Czyli funkcja \(f(x)\) jest rosnąca na przedziale \(\left(0,\frac{\sqrt{3}}{2}\right)\) i malejąca na przedziale \(\left(\frac{\sqrt{3}}{2},\sqrt{2}\right)\).

W punkcie \(x=\frac{\sqrt{3}}{2}\) funkcja \(f(x)\) osiąga maksimum równe \(b=1/2\).

funkcja.subs({x:sqrt(3)/2})
1/2

Wizualizacja pudełka ograniczającego kształt lemniskaty

#rysowanie lemniskaty ze zmodyfikowaną skalą
pup=plot(f,(x,-sqrt(2),sqrt(2)),xmin=-sqrt(2)-0.2,xmax=sqrt(2)+0.2,ticks=[[-sqrt(2),-sqrt(2)/2,0,sqrt(2)/2,sqrt(2)],[-1/2,0,1/2]],tick_formatter=[sqrt(2)/2,1/2])
pdown=plot(-f,(x,-sqrt(2),sqrt(2)),ticks=[[-sqrt(3)/2,0,sqrt(3)/2],[-1/2,0,1/2]])
lemniskata=pdown+pup

#pudełko dookoła lemniskaty
pudelko=plot(1/2,(x,-sqrt(2),sqrt(2)),color='red',fill='axis',fillcolor='green',fillalpha=0.1)+line([(-sqrt(2),-1/2),(-sqrt(2),1/2)],color='red')
pudelko+=plot(-1/2,(x,-sqrt(2),sqrt(2)),color='red')+line([(sqrt(2),-1/2),(sqrt(2),1/2)],color='red')

#linie maksimów (przerywane, zielone)
linie=line([(sqrt(3)/2,0),(sqrt(3)/2,1/2)],linestyle="--",color="green")
linie+=line([(-sqrt(3)/2,0),(-sqrt(3)/2,1/2)],linestyle="--",color="green")

#dodatkowe etykiety
teksty=text("$\\frac{\\sqrt{3}}{2}$",(sqrt(3)/2+0.05,-0.05),color='green')
teksty+=text("$\\frac{-\\sqrt{3}}{2}$",(-sqrt(3)/2-0.05,-0.05),color='green')

#końcowy wykres
show(pudelko+lemniskata+teksty+linie,figsize=[6,6],aspect_ratio=1)
_images/85e0a9c29f9bcc04933b44d1645a6a7a2b530d71392a75e35e5479fd9b1051e0.png

Teraz możemy obliczyć wartość pola wewnątrz figury jako sumę czterech całek (każda odpowiada jednej ćwiartce figury).

plg=plot(f,(x,-sqrt(2),0),fill='min',fillcolor='blue', fillalpha=.2)
pld=plot(-f,(x,-sqrt(2),0),fill='max',fillcolor='red', fillalpha=.2)
ppg=plot(f,(x,0,sqrt(2)),fill='min',fillcolor='green', fillalpha=.2)
ppd=plot(-f,(x,0,sqrt(2)),fill='max',fillcolor='yellow', fillalpha=.2)
show(plg+pld+ppg+ppd,figsize=[4,4],aspect_ratio=1)
_images/d198adfe27f46426ab57e256fa69a48046a3efbcd186119727cd64056aaf79ed.png
#Pole wewnątrz lemniskaty
2*integrate(funkcja,(x,-sqrt(2),sqrt(2))).n()
2.0000000070692856
#szybsza metoda numerycznego całkowania z kontrolą błędu (wartość całki, błąd)
integral_numerical(2*funkcja,-sqrt(2),sqrt(2),max_points=100)
(2.0000000070692856, 1.4825283125400555e-06)

Hmm, być może wartość pola wynosi \(2\), więc wartość całki z \(f(x)\) wynosi \(1\) w przedziale \((-\sqrt{2},\sqrt{2})\). Sprawdźmy to !

Hipoteza#

\[P=\int_{0}^{\sqrt{2}}\sqrt{\sqrt{4x^2 + 1}-x^2 - 1}\: dx = \frac{1}{2}\]

Wyznaczymy najpierw parametryzację promieniem i kątem \((r,t)\) dla naszego równania krzywej Bernoulliego.

Korzystamy ze standardowej zamiany zmiennych

\[x=r\cos(t),\quad y=r\sin(t)\]
r,t=var('r,t')
((x**2+y**2)**2-2*(x**2-y**2)).subs({x:r*cos(t),y:r*sin(t)}).simplify_full().factor()
(r^2 + 4*sin(t)^2 - 2)*r^2

Zatem równanie uwikłane lemniskaty we współrzędnych biegunowych \((r,t)\) jest postaci

\[r^2+4\sin(t)^2-2=0\]
solve(r^2 + 4*sin(t)^2 - 2==0,r)
[r == -sqrt(-4*sin(t)^2 + 2), r == sqrt(-4*sin(t)^2 + 2)]

Zatem możemy sparametryzować naszą krzywą następującymi równaniami

\(x=r \cos(t), \quad y=r \sin(t)\), gdzie

\(r=\pm \sqrt{2-4 \sin(t)^2}=\pm \sqrt{2\cos(t)}=\pm \sqrt{4\cos(t)^2-2}\)

show(parametric_plot((sqrt(2*cos(2*t))*cos(t),sqrt(2*cos(2*t))*sin(t)),(t,0,pi/4) ,figsize=[3,3]))
_images/b740f958e81c92725b3ed0db98ded3efa05027dc2891ee80c4c5ed4ea3f39138.png
show(parametric_plot((-sqrt(2*cos(2*t))*cos(t),-sqrt(2*cos(2*t))*sin(t)),(t,-pi/4,pi/4),figsize=[3,3]))
_images/da5ec0321679e0ed6c3192069f4b039d946abfb5b7e07bdf34ab4fe930a4f827.png

Świetnie, zatem możemy naszą całkę \(P\) przetransformować do postaci, gdzie będzie zależała wyłącznie od zmiennej \(t\).

f1=funkcja.subs({x:sqrt(4*cos(t)**2-2)*cos(t)})
d1=x.subs({x:sqrt(4*cos(t)**2-2)*cos(t)}).derivative(t)
f1=f1.simplify_full()
pretty_print(f1)

Zauważmy, że

\[16\cos(t)^4 - 8\cos(t)^2 + 1 = (4\cos(t)^2 - 1)^2\]

oraz

\(4\cos(t)^2 - 1\geq 0\) skoro \(t\in(0,\frac{\pi}{4})\)

f1=sqrt(-4*cos(t)**4+2*cos(t)**2+(4*cos(t)**2-1)-1)
pretty_print(f1.simplify_full())

To wyrażenie uprościmy dalej

$\(\sqrt{-4\sin(t)^4+2\sin(t)^2} = \sin(t)\sqrt{2(1-2\sin(t)^2)}=\sin(t)\sqrt{4\cos(t)^2-2}\)\( na mocy naszych założeń o \)t$.

f1=(sin(t)*sqrt(4*cos(t)**2-2))
d1=d1.factor()
pretty_print((f1*d1).simplify().expand().simplify_full())
(f1*d1).simplify().expand().simplify_full()
-2*(4*cos(t)^2 - 1)*sin(t)^2

Zatem metodą dokładną pole wewnątrz lemniskaty jest czterokrotnym polem jednego parametryzowanego kawałka \(P\). Parametryzacja jest zgodna z orientacją osi, gdy \(t\) maleje od \(\frac{\pi}{4}\) do \(0\). Czyli

\(P = \int_{x=0}^{x=\sqrt{2}} \sqrt{\sqrt{4x^2 + 1}-x^2-1}\: dx=\int_{t=\pi/4}^{t=0}-2(4\cos(t)^2-1)\cdot \sin(t)^2 dt\)

integrate(-2*(4*cos(t)^2 - 1)*sin(t)^2,(t,pi/4,0))
1/2

Problem#

Czy możemy obliczyć przybliżenie pola lemniskaty za pomocą metody nieanalitycznej?

Metoda numeryczna 1:#

Wygenerujemy zbiór punktów na płaszczyźnie w obszarze, w którym mieści się cała lemniskata.

Nasza lemniskata styka się z bokami prostokąta \(A=[-\sqrt{2},\sqrt{2}]\times [-1/2,1/2]\).

Pole tego prostokąta wynosi \(2\sqrt{2}\). Wylosujemy punkty w obu przedziałach (jednostajnie) i zobaczymy ile ich potrzeba, aby uzyskać rozsądne przybliżenie.

UWAGA: W tym rozwiązaniu zastosujemy generator liczb pseudolosowych, który produkuje dla nas liczby z przedziału \([0,1]\) z rozkładem jednostajnym.

r=random() #zmienna o rozkładzie jednostajnym i wartościac w przedziale [0,1]
show(list_plot([random() for i in range(0,100)],figsize=[3,3]))
_images/e87ee235f75a4c0fbed1d21fc610ba2430939fda527519197c9c9f81255fd8df.png

Generujemy losowy punkt w prostokącie \(A\)

def LosowyPunkt():
    x=sqrt(2).n()*(2*random()-1)
    y=random()-1/2
    return (x,y)
li=[point(LosowyPunkt()) for _ in range(0,100)]
show(sum(li),figsize=[3,3])
_images/310aaf75f0773f1308666bb79a65eaf8e10359f7f159e4d8a638199501d9f4ad.png

Teraz potrzebujemy funkcji, która wskaże czy zadany punkt znajduje się we wnętrzu lemniskaty lub nie.

def CzyWewnatrzLemniskaty(punkt):
    x,y=punkt
    f=sqrt(-x^2 + sqrt(4*x^2 + 1) - 1)
    if abs(y)<=f:
        return True
    else:
        return False
CzyWewnatrzLemniskaty((1/4,1/8))
True
CzyWewnatrzLemniskaty((0,1))
False
lem=plot(f,(x,-sqrt(2),sqrt(2)),fill='min',fillcolor='yellow', fillalpha=.2,figsize=[4,4],aspect_ratio=1,ticks=[[],[]])
lem+=plot(-f,(x,-sqrt(2),sqrt(2)),fill='max',fillcolor='yellow', fillalpha=.2)
pt=point((1/4,1/8))+point((0,1))
teksty=text("(1/4,1/8)",(1/4+0.4,1/8))+text("(0,1)",(+0.2,1-0.1))
teksty+=text("Lemniskata $(x^2+y^2)^2=2(x^2-y^2)$", (0.0,1.3), background_color=(1,1,0.8),fontsize=8)

show(lem+pt+teksty)
_images/f97b933baf56f3594ae4d913fad20a36341533f8da91547228db33132d594221.png

Obliczanie pola polega na wygenerowaniu zestawu próbek punktów, dla których sprawdzamy czy należą do wnętrza lemniskaty.

def PoleLemniskaty(proba):
    traf=0
    for _ in range(0,proba):
        if CzyWewnatrzLemniskaty(LosowyPunkt()):
            traf+=1
    return traf/proba*(2*sqrt(2).n()) #skalujemy ze względu na rozmiar prostokąta A
show(list_plot([PoleLemniskaty(1000*k) for k in range(1,10)],figsize=[3,3]))
_images/fd757f77a7e4dcce1b7942461e11d6691b937cba01cc2952f93c8a04a427b2d6.png

Przy wyborze niewielkiej liczby punktów próbkujących dokładność jest niezadowalająca. Jedną z przyczyn kłopotów z dobrym przybliżeniem jest też jakość próbek losowych pochodzących z funkcji random.

PoleLemniskaty(5)
2.82842712474619
PoleLemniskaty(500)
2.00818325856980
PoleLemniskaty(50000)
2.00224356160783

Metoda numeryczna 2#

Całkowanie z wykorzystaniem spacerów losowych

W ostatniej części postawimy sobie ambitniejsze zadanie. Zakładamy na początek, że mamy do dyspozycji tylko prosty generator bitowy genbit (który zwraca wartości \(0\) i \(1\) z prawdopodobieństwem bliskim \(50\%\)).

Skonstruujemy na bazie generatora bitowego zmienną losową step, która z prawdopodobieństwem \(1/4\) zwraca wartość \(-1\), z prawdopodobieństwem \(1/2\) zwraca wartość \(0\) i z prawdopodobieństwem \(1/4\) zwraca wartość \(1\).

Zmienna \(K\) może posłużyć do konstrukcji spaceru losowego po prostej, w którym połowę czasu spędzamy stojąc w miejscu.

def genbit():
    return choice([0,1])
proba=100000
probka=[genbit() for _ in range(0,proba)]

print("Częstość zer = "+str(probka.count(0)/proba*1.0))
print("Częstość jedynek = "+str(probka.count(1)/proba*1.0))
Częstość zer = 0.501960000000000
Częstość jedynek = 0.498040000000000
def step():
    go=genbit()
    if go==1:
        left=genbit()
        if left==1:
            return -1
        else:
            return 1
    else:
        return 0

Zbadajmy numerycznie rozkłąd zmiennej losowej step

proba=1000
li=[step() for i in range(0,proba)]
le=li.count(-1)
st=li.count(0)
ri=li.count(1)
ll=len(li)
print(''.join(["P(step={})={}\n".format(x[0],x[1]*1.0/ll) for x in [(-1,le),(0,st),(1,ri)]]))
P(step=-1)=0.240000000000000
P(step=0)=0.508000000000000
P(step=1)=0.252000000000000

Dla ustalonej pozycji początkowej \(\{-1,0,1\}\) konstruujemy dodatkową funkcję, która zwróci nam losową pozycję (stan) po zastosowaniu zmiennej step.

states=[-1,0,1]
def RandomStep(init):
    s=step()
    if s==0:
        return init
    else:
        return states[(init+s+1)%3]
proba=1000
li=[RandomStep(1) for i in range(0,proba)]
le=li.count(-1)
st=li.count(0)
ri=li.count(1)
ll=len(li)
print(''.join(["P(step={})={}\n".format(x[0],x[1]*1.0/ll) for x in [(-1,le),(0,st),(1,ri)]]))
P(step=-1)=0.253000000000000
P(step=0)=0.255000000000000
P(step=1)=0.492000000000000

Teraz naszym celem jest skonstruowanie zmiennej losowej, która przyjmuje jeden ze stanów \(\{-1,0,1\}\) z prawdopodobieństwem \(1/3\). Nie możemy bezpośrednio do tego celu wykorzystać naszej poprzedniej zmiennej, ale korzystając z niej możemy skonstruować łańcuch Markowa, który będzie miał tę cechę, że jego rozkład graniczny będzie jednostajny.

Nasz łańcuch Markowa jest aperiodyczny i nierozkładalny, zatem posiada jedyny stan stacjonarny, do którego zbiega każdy stan początkowy. Stan stacjonarny odpowiada wektorowi własnemu (znormalizowanemu) podporządkowanemu wartości własnej \(1\).

m=matrix([[1/2,1/4,1/4],[1/4,1/2,1/4],[1/4,1/4,1/2]])
m.eigenvalues()
[1, 1/4, 1/4]
#graf łańcucha Markowa
g = DiGraph({-1: [-1,0,1], 0: [-1,0,1],1: [-1,0,1]}); 
show(g)
_images/20e8775314d0e4f2757c553b7fd35ecf92909820358b729cea67e44280e8d38b.png
pi_vec=m.eigenvectors_right()[0][1][0]*1/3
m*pi_vec==pi_vec
True

Tempo zbieżności do rozkładu stacjonarnego jest dość duże w tym przypadku

m^10*1.0
[0.333333969116211 0.333333015441895 0.333333015441895]
[0.333333015441895 0.333333969116211 0.333333015441895]
[0.333333015441895 0.333333015441895 0.333333969116211]

Poniżej konstruujemy funkcję, która przy zadanym stanie początkowym i odpowiednim czasie mieszania mixing generuje wartości zgodne z łańcuchem Markowa opisanym powyżej. Wykorzystanie yield pozwala nam utworzyć jeden proces, który będzie można wykorzystywać we wszystkich funkcjach.

def MarkovChain(seed,mixing):
    state=seed
    #mieszanie początkowe dochodzące do stanu jednostajnego rozkładu
    for _ in range(0,mixing):
        state=RandomStep(state)
    while true:
        state=RandomStep(state)
        yield state

Sprawdzimy numerycznie własności naszego łańcucha Markowa

m1=MarkovChain(-1,500) #start w stanie -1 i "wygrzewanie" łańcucha w 500 krokach startowych 
                       #(aby zgubić początkowy rozkład)
le=0 #zliczanie wartości -1
st=0 #zliczanie wartości 0
ri=0 #zliczanie wartości 1
samp=50000 #rozmiar próby
for _ in range(0,samp):
    v=next(m1)
    if v==-1:
        le+=1
    else:
        if v==0:
            st+=1
        else:
            ri+=1
[x*1.0/samp for x in [le,st,ri]]
[0.336800000000000, 0.330300000000000, 0.332900000000000]

Na bazie łańcucha Markowa MarkovChain możemy utworzyć spacer losowy po grafie odwzorowującym punkty o współrzędnych całkowitych na płaszczyźnie. Taki spacer posłuży nam do generowania próbek losowych punktów na płaszczyźnie (po normalizacji).

def RandomWalk(point):
    x,y=point
    mx=MarkovChain(0,500)
    my=MarkovChain(0,500)
    lipt=[(x,y)]
    while true:
        x+=next(mx) #modyfikacja zmiennej x za pomocą łańcucha mx
        y+=next(my) #modyfikacja zmiennej y za pomocą łańcucha my
        yield (x,y) #dodanie kolejnego punktu w spacerze 

Zobaczmy jak “wędruje” punkt po płaszczyźnie w spacerze losowym zadanym powyżej.

rw=RandomWalk((0,0))
lin=[(0,0)]

for _ in range(0,1000):
    pp=next(rw)
    lin.append(pp)
    
#show(pt)
show(line(lin),aspect_ratio=1,figsize=[4,4])
_images/67ff6495b272d2505517d139b7755108e9d82c17b7184eed0ca29dea23683b82.png

W praktyce interesuje nas spacer nie po nieograniczonej płaszczyźnie, ale po organiczonym jej kawałku. Aby zastosować taki efekt, utożsamiamy brzegi pewnego kwadratu, aby uzyskać topologiczny model torusa, na którym realizuje się nasz spacer losowy.

def RandomWalkWithBounds(point,size):
    x,y=point
    mx=MarkovChain(0,500)
    my=MarkovChain(0,500)
    lipt=[(x,y)]
    while true:
        x+=next(mx) #modyfikacja zmiennej x za pomocą łańcucha mx
        x=x%size
        y+=next(my) #modyfikacja zmiennej y za pomocą łańcucha my
        y=y%size
        yield (x,y) #dodanie kolejnego punktu w spacerze 

Poniżej prześledzimy na mapie częstości (im ciemniejszy piksel tym więcej odwiedzin) jak często dany punkt jest odwiedzany w spacerze losowym. W granicy każdy punkt powinien być odwiedzany z tą samą częstotliwością.

rozmiar=30
rw=RandomWalkWithBounds((rozmiar//2,rozmiar//2),rozmiar)
li1=[]
for _ in range(0,50000):
    li1.append(next(rw))

freq={(i,j):(li1.count((i,j))) for i in range(0,rozmiar) for j in range(0,rozmiar)}
m=max([freq[x] for x in freq])*1.0
M=matrix(RR,rozmiar,rozmiar,0)
for el in freq:
    M[el]=freq[el]/m
    
show(matrix_plot(M),figsize=[3,3])
_images/0d71d64b9d7717c5f6b153a9bc5eff2b802339fd7debca9aea436fa04ed1259b.png

Teraz możemy zdefiniować funkcję, która przy zadanym kształcie będzie zliczała odwiedziny.

def ShapeFunction(x,y,S):
    if S(x,y): #S jest warunkiem bycia wewnątrz kształtu
        return 1
    else:
        return 0
    
S1=(lambda x,y: y**2<=sqrt(4*x**2 + 1)-x**2 - 1) # wnętrze lemniskaty

I jeszcze jedna techniczna funkcja skalująca.

def MapFunction(a,b,xbound,xsize,ybound,ysize): #(a,b) w [0,1]^2
    return xbound+a*xsize,ybound+b*ysize

Teraz możemy już przygotować schemat całkowania metodą Monte Carlo z wykorzystaniem spaceru losowego. Po zainicjalizowaniu spaceru zliczamy te punkty wewnątrz kształtu, które odwiedziliśmy w trakcie spaceru. Zgodnie ze spodziewanym rozkładem jednostajnym zliczamy częstość odwiedzonych punktów w stosunku do wszystkich i mnożąc przez rozmiar próbkowanej przestrzeni.

def CalkowanieMarkowa(start,steps,size,S,xbound,xsize,ybound,ysize):
    s=0
    rw=RandomWalkWithBounds(start,size)
    for _ in range(0,steps):
        a,b=next(rw)
        x,y=MapFunction(a*1.0/size,b*1.0/size,xbound,xsize,ybound,ysize)
        s+=ShapeFunction(x,y,S)
    return s*(xsize)*(ysize)/steps

Przy nawet stosunkowo małych rozdzielczościach próbkowania (przykład poniżej dla 30 i 40 punktów w każdym z kierunków) i niezbyt dużej liczbie iteracji, otrzymujemy zadowalające przybliżenie wyniku.

CalkowanieMarkowa((15,15),5,30,S1,-sqrt(2).n(),2*sqrt(2).n(),-1/2,1)
2.82842712474619
CalkowanieMarkowa((15,15),500,30,S1,-sqrt(2).n(),2*sqrt(2).n(),-1/2,1)
1.94030100757589
CalkowanieMarkowa((15,15),50000,30,S1,-sqrt(2).n(),2*sqrt(2).n(),-1/2,1)
2.00659933937994

Przy rozdzielczości \(40\times 40\) punktów kraty i \(2000\) iteracji otrzymujemy, że średnia wartość obliczonego metodą Monte Carlo pola ma błąd poniżej 0.05, a mediana przypada na wartość, której odchylenie również mieści się w zakresie około \(0.01\).

Na wykresie poniżej prezentujemy rozkład wartości pola dla 100 prób wykonanych obliczeń.

li=[CalkowanieMarkowa((randint(0,40),randint(0,40)),2000,40,S1,-sqrt(2).n(),2*sqrt(2).n(),-1/2,1)*1.0 for i in [1..100]] 
print(sum(li)/len(li)) #wartość średnia po wielu iteracjach
li.sort()
print(li[floor(len(li)/2)]) #wartość środkowa (mediana)
show(list_plot(li))
1.99339058470737
2.01808275350641
_images/d9ef9cf6a134fc25dfccafa4391b13b5441b250ece2a7a51f5990c0b882e3352.png