#!/usr/bin/env python # coding: utf-8 # # ODEs: Higher order IVP # * EE and IE are simple, but low order: globally first order. # * To get high accuracy requires many more steps: smaller $\Delta t$. # * Higher order methods give the same accuracy for larger step sizes. # # # # ## Second order methods # # ODE: # $$\frac{dy}{dt} = f(y,t),$$ # $$y(t=0) = y_0.$$ # # # ### Modified midpoint (MM) method # # * EE takes a step from $k$ to $k+1$ using the slope at point $k$. # * IE takes a step from $k$ to $k+1$ using the slope at point $k+1$. # * MM takes a step from $k$ to $k+1$ using the slope at point $k+1/2$. # * To get to $k+1/2$ so that we can evaluate the slope there, take a trial EE step to that point # # # $$y_{k+1/2} = y_k + \frac{\Delta t}{2}f(y_k,t_k).$$ # $$y_{k+1} = y_k + \Delta tf(y_{k+1/2},t_{k+1/2}).$$ # # # * So, on each step we first evaluate $y_{k+1/2}$ using EE for a half step. Then we take the real full step using the slope at $y_{k+1/2}$. # #### Taylor series representation # * Write two Taylor series approximations for $y_{k+1}$ and $y_k$ centered on $y_{k+1/2}$. # $$y_{k+1} = y_{k+1/2} + y^{\prime}_{k+1/2}\frac{\Delta t}{2} + \frac{1}{2}y^{\prime\prime}_{k+1/2}\left(\frac{\Delta t}{2}\right)^2+\frac{1}{6}y^{\prime\prime\prime}_{k+1/2}\left(\frac{\Delta t}{2}\right)^3\ldots,$$ # $$y_{k} = y_{k+1/2} - y^{\prime}_{k+1/2}\frac{\Delta t}{2} + \frac{1}{2}y^{\prime\prime}_{k+1/2}\left(\frac{\Delta t}{2}\right)^2-\frac{1}{6}y^{\prime\prime\prime}_{k+1/2}\left(\frac{\Delta t}{2}\right)^3\ldots,$$ # * Subtract the second equation from the first and solve for $y_{k+1}$. # $$y_{k+1} = y_k + \Delta t y^{\prime}_{k+1/2} + \mathcal{O}(\Delta t^3).$$ # * The method is globally $\mathcal{O}(\Delta t^2)$. # * Note, $y^{\prime}_{k+1/2} \equiv f(y_{k+1/2},t_{k+1/2}).$ # * Note, this analysis was done using $y_{k+1/2}$, but the actual method doesn't use $y_{k+1/2}$, it uses the EE approximation to this value, so we really have # $$y_{k+1} = y_k + \Delta t (y^{\prime}_{k+1/2}+\mathcal{O}(\Delta t^2)) + \mathcal{O}(\Delta t^3),$$ # where the $\mathcal{O}(\Delta t^2)$ term is the local error of the EE method. When we expand this we get # $$y_{k+1} = y_k + \Delta t y^{\prime}_{k+1/2}+\mathcal{O}(\Delta t^3) + \mathcal{O}(\Delta t^3) = y_k + \Delta ty^{\prime}_{k+1/2}+\mathcal{O}(\Delta t^3),$$ # so there is no overall change in the global order of the method. # ### Implicit trapazoid (IT) method # * From the MM method: # $$ y_{k+1} = y_k + \Delta t y^{\prime}_{k+1/2} = y_k + \Delta t f_{k+1/2}.$$ # * Evaluate # $$f_{k+1/2} = \frac{1}{2}(f_k + f_{k+1}).$$ # * Hence, # # # $$y_{k+1} = y_k + \Delta t\frac{f(y_k,t_k) + f(y_{k+1},t_{k+1})}{2}.$$ # # # * This method is globally $\mathcal{O}(\Delta t^2)$ accurate. # * Note how and why the method is implicit. # * Note, this method can also be written as EE from $k$ to $k+1/2$ followed by IE from $k+1/2$ to $k+1$: # $$y_{k+1/2} = y_k + \frac{\Delta t}{2}f_k,$$ # $$y_{k+1} = y_{k+1/2} + \frac{\Delta t}{2}f_{k+1}.$$ # # # ### Modified Euler (ME) method # * This is like the IT method, but get $f_{k+1}$ from a full EE step: # # # $$ y^*_{k+1} = y_k + \Delta t f_k,$$ # $$ y_{k+1} = y_k + \Delta t \frac{f(y_k,t_k) + f(y_{k+1}^*,t_{k+1})}{2}.$$ # # # * This is a two-step **predictor-corrector** method. We predict the answer using EE in the first step, then using this result, we improve (correct) the answer in the second step. # * This method is globally $\mathcal{O}(\Delta t^2)$ accurate. # ## Runge Kutta (RK) methods # # \begin{align*} # y_{k+1} = y_k + &hS, \\ # & S = c_1S_1 + c_2S_2 + c_3S_3 + \ldots. # \end{align*} # # * Here, let $h\equiv\Delta t$ # * $S$ is a slope $dy/dt$, evaluated from $f(y,t)$ or from a combination of $f(y,t)$ evaluated at various points $y$ in different ways. # * The $c_i$ above are coefficients of the linear combination. # * Here, we will ignore the time dependence of $f$ for simplicity and clarity. # ### Second order # \begin{align*} # y_{k+1} = &y_k + h(c_1S_1 + c_2S_2), \\ # &S_1 = f(y_k), \\ # &S_2 = f(y_k + \beta hS_1). # \end{align*} # * So, we are taking linear combinations of $h$ times the slopes evaluated at different points. # * Note in the second equation we evaluate the slope at a later time. We get to that time by taking an Euler-like step $y_k+\beta hS_1$. # * Combine the above equations: # # # $$ y_{k+1} = y_k + c_1hf(y_k) + c_2hf(y_k+\beta hS_1).$$ # # # * Again (for slides display): # # # $$ y_{k+1} = y_k + c_1hf(y_k) + c_2hf(y_k+\beta hS_1).$$ # # # * This equation contains parts at point $k$ and a part at a later time, $f(y_k+\beta hS_1)$. # * Note that $\beta hS_1 = \beta hf_k$. # * We can write all parts at point $k$ by expanding $f(y_k+\beta hS_1)$ in a Taylor series about point $k$: # # \begin{align} # f(y_k+\beta\Delta y_1) &= f_k + \left.\frac{\partial f}{\partial y}\right|_k\beta hS_1 + \mathcal{O}(h^2), \\ # &= f_k + \beta h f_{y,k} f_k + \mathcal{O}(h^2),\\ # \end{align} # # where $f_{y,k}$ is the derivative of $f$ with respect to $y$ evaluated at point $k$. # # * Hence, # # # $$ y_{k+1} = y_k + f_k h(c_1+c_2) + \beta h^2c_2f_{y,k}f_k + \mathcal{O}(h^3).$$ # # # # * Now, we will take a Taylor series of $y_{k+1}$ and compare it with the above green equation. # # \begin{align} # y_{k+1} &= y_k + y_k^{\prime}h + \frac{1}{2} y_k^{\prime\prime}h^2 + \mathcal{O}(h^3),\\ # &= y_k + f_kh + \frac{1}{2}h^2(y^{\prime})^{\prime}_k + \mathcal{O}(h^3). # \end{align} # # * Now, # # $$(y^{\prime})^{\prime} = \left.\frac{\partial f}{\partial t}\right|_k = \left.\frac{\partial f}{\partial y}\frac{dy}{dt}\right|_k = f_{y,k}f_k.$$ # # * Hence: # # # $$y_{k+1} = y_k + f_kh + \frac{1}{2}h^2f_{y,k}f_k + \mathcal{O}(h^3).$$ # # # * Again, the green equation above is: # # # $$ y_{k+1} = y_k + f_k h(c_1+c_2) + \beta h^2c_2f_{y,k}f_k + \mathcal{O}(h^3).$$ # # # * Comparing the green and red equations imples: # # # $$ (c_1+c_2) = 1, $$ # $$\beta c_2 = \frac{1}{2}.$$ # # # * So, in choosing our second order RK method, we have one degree of freedom. Choose, say, $c_1$, then $c_2 = 1-c_1$, and then $\beta = 1/2c_2.$ # * There are an infinite number of possible second order methods. # ### Example # Recall our 2nd order method: # \begin{align*} # y_{k+1} = &y_k + h(c_1S_1 + c_2S_2), \\ # &S_1 = f(y_k), \\ # &S_2 = f(y_k + \beta hS_1). # \end{align*} # # * Let $c_1 = 1/2$, then $c_2=1/2$ and $\beta = 1$. # $$y_{k+1} = y_k + \frac{h}{2}(f(y_k) + f(y_k+hf(y_k))$$ # * This is the ME method. # ### Fourth order RK method (*The* RK method) # # \begin{align*} # y_{k+1} &= y_k + h\left(\frac{1}{6}S_1 + \frac{2}{6}S_2 + \frac{2}{6}S_3 + \frac{1}{6}S_4\right),\\ # & S_1 = f(y_k), \\ # & S_2 = f(y_k+\frac{h}{2}S_1), \\ # & S_3 = f(y_k+\frac{h}{2}S_2), \\ # & S_4 = f(y_k+hS_3). # \end{align*} # # # ### Approach # * Evaluate the slope $S_1$ at $y_k$. # * Take a half step EE from point $k$ to $k+1/2$ using slope $S_1$. # * Evaluate $S_2$ here. # * Take a half step EE from point $k$ to $k+1/2$ using slope $S_2$. # * Evaluate $S_3$ here. # * Take a full step EE from point $k$ to $k+1$ using slope $S_3$. # * Evaluate $S_4$ here. # * Take the final full step using a linear combination of the slopes: # $$y_{k+1} = y_k + \frac{h}{6}(S_1 + 2S_2 + 2S_3+ S_4).$$ # * Note how values of $y$ at intermediate step are used to compute slopes for following steps. # * For $n^{th}$ order with $n\le 4$ $\rightarrow$ $n$ function evaluations are required. # * For $>4^{th}$ order $\rightarrow$ more than $n$ function evaluations required. (Storage costs too). # * $\rightarrow$ the $4^{th}$ order method gives the most bang for the buck. # ### RK4 code # * Solve $dy/dt = -y$ from $t=0$ to $t=5$ with RK4 and EE. Use 10 steps, so $\Delta t = 0.5$. Let $y_0=1$. # In[1]: import numpy as np import matplotlib.pyplot as plt get_ipython().run_line_magic('matplotlib', 'inline') # In[2]: def odeRK4(f, y0, t): ns = len(t)-1 y = np.zeros(ns+1) y[0] = y0 for k in range(ns): h = t[k+1]-t[k] S1 = f(y[k],t[k]) S2 = f(y[k]+0.5*h*S1, t[k]+0.5*h) S3 = f(y[k]+0.5*h*S2, t[k]+0.5*h) S4 = f(y[k]+ h*S3, t[k]+ h) y[k+1] = y[k] + h/6*(S1 + 2*S2 + 2*S3 + S4) return y # In[4]: def odeEE(f, y0, t): ns = len(t)-1 y = np.zeros(ns+1) y[0] = y0 for k in range(ns): h = t[k+1]-t[k] y[k+1] = y[k] + h*f(y[k],t[k]) return y # In[5]: def f(y, t): return -y #---------------------- y0 = 1.0 tend = 5.0 t = np.linspace(0,tend,11) #---------------------- y_RK4 = odeRK4(f,y0,t) y_EE = odeEE(f,y0,t) tt = np.linspace(0,tend,100) y_Exact = np.exp(-tt) #---------------------- # In[6]: plt.rc('font', size=14) plt.plot(tt,y_Exact,'k--') plt.plot(t, y_RK4, 'o') plt.plot(t, y_EE, '^') plt.legend(['exact', 'RK4', 'EE'], frameon=False) plt.ylabel('y') plt.xlabel('t'); # In[ ]: