Reaction-diffusion equations ============================ Up to now, we have analysed models of population dynamics and kinetics of chemical reactions without spread in space or area. We have been interested in behavior of the system in time and not in space. It means that modeling is based on ordinary differential equations (derivatives are with repect to time. Changes in space are described by derivatives with respect to coordinates. The we obtain partial differential equations. Mathematical analysis of such equations is much more difficult. Today, an alternative is numerical analysis with the help of computers. Example: diffusion ------------------ If we put a drop of dye into water in a glass, we can observe spread of color and the blop will increases in time. After some time water in glass will be colorful. This process of spread of dye is called diffusion. Particle of dye in water moves in a very irregular way. This molecular motion is called Brownian motion and the particle of dye are named Brownian particles. Brownian particles suspended in fluid collides with particles of fluid and the position of the Brownian particle fluctuates in a chaotic (random) way. In the mathematical formalism, diffusion can be taken into account by adding the Laplacian to evolution equations: .. math:: \frac{\partial c(\vec r, t)}{\partial t} = D \Delta c(\vec r, t) = D \left[\frac{\partial^2 }{\partial x^2} + \frac{\partial^2 }{\partial y^2} + \frac{\partial ^2}{\partial z^2} \right] c(\vec r, t) = \\ D \left[\frac{\partial^2 c(\vec r, t)}{\partial x^2} + \frac{\partial^2 c(\vec r, t)}{\partial y^2} + \frac{\partial ^2 c(\vec r, t)}{\partial z^2} \right] where :math:`c(\vec r, t)` is the concentration of the dye particles at time :math:`t` in the point of space of coordinates :math:`\vec r = (x, y, z)`. Let us note that the diffusion process is described by the second-order derivatives with respect to the spatial variables. The above equation is named the diffusion equation. For the initial concentration :math:`c(x, y, z, 0)` of particles which all are in the point :math:`(x,y,z)=(0,0,0)` the solution of the diffusion equation has the form .. math:: c(x, y, z, t) = [4\pi Dt]^{-3/2} \; exp\left[-\frac{x^2+y^2+z^2}{4Dt}\right] This equations can be solved by e.g. Fourier transform method. It is a Gaussian function of the bell shape. Its half-width is proportional to :math:`D t`. It means that if time :math:`t` increases, the half-with increases and this property describes spread of diffusiong particles. For the proces only in one dimension, e.g. along the :math:`OX` axis, the corresponding diffusion equation has the form .. math:: \frac{\partial c(x, t)}{\partial t} = D \frac{\partial^2 }{\partial x^2} c(x, t) = D \frac{\partial^2 c(x, t) }{\partial x^2} Its solution reads .. math:: c(x, t) = [4\pi Dt]^{-1/2} \; exp\left[-\frac{x^2}{4Dt}\right] If we want to take into account the spatial spread of population (or chemical products), we can formulate the following method: .. admonition:: Recipe One has to add the Laplace operator to evolution equations If spread is in 2-dimensional region, we should add the term: .. math:: + D \left[ \frac{\partial^2 }{\partial x^2} + \frac{\partial^2 }{\partial y^2} \right] c(x, y, t) Solution of the 2-dimensional diffusion equation is the Gaussian function on the form: .. math:: c(x, y, t) = [4\pi Dt]^{-2/2} \; \exp\left[-\frac{x^2+y^2}{4Dt}\right] =\frac{1}{4\pi Dt} \; \exp\left[-\frac{x^2+y^2}{4Dt}\right] .. code:: ipython2 var('x,y,t') c(x,y,t) = 1/(4*pi*t)*exp(-(x^2+y^2)/(4*t)) plts = [ plot3d(c(x,y,t), (x,-4,4), (y,0,8), rgbcolor=hue(t/2.0)) for t in [0.5,1.0,1.5,2.0]] @interact def _(viewer=['tachyon','jmol','canvas3d']): show(sum(plts),viewer=viewer) .. code:: ipython2 def plt_t(t,**kwargs): cmax = c(0,0,t) return contour_plot( c(x,y,t) ,(x,-5,5),(y,-5,5),contours=srange(0.01,cmax,cmax/20.0), cmap='spectral',figsize=(3,3),**kwargs) .. code:: ipython2 times = [.1,0.4,0.9] for t in times: plt_t(t).show(title="t=%0.3f"%t) .. image:: output_3_0.png .. image:: output_3_1.png .. image:: output_3_2.png All considered models (Malthus, Verhuslt, chemical reactions, Lotka-Volterra, May, …) can be extended includin spatial chages. For example, if an evolution equation has the form .. math:: \frac{dN(t)}{dt} = F(N(t)) then its generalization with diffusion has the form .. math:: \frac{\partial N(\vec r, t)}{\partial t} = F(N(\vec r, t) ) + D \Delta N(\vec r, t) where the Laplacian can be taken in 1, 2 or 3 dimensions. **This type of equations are called reaction-diffusion equations** Below, we consider two examples: the Malthus model and the Verhulst model with diffusion. The Malthus model with migration: The Skellam equation ====================================================== The standard Malthus model is the simplest model of growth or dacay processes. It is defined by the equation .. math:: \frac{dN(t)}{dt} = k N(t), \quad k - real If the population can randomly moves on the plane :math:`XY` then the extended equation takes the form .. math:: \frac{dN(x, y, t)}{dt} = k N(x, y, t) + D \left[ \frac{\partial^2 }{\partial x^2} + \frac{\partial^2 }{\partial y^2} \right] N(x, y, t) where now :math:`N(x, y, t)` is the concentration (density, i.e. a number of individuals per unit area) in the instant :math:`t` in the point on the :math:`XY` plane with the coordinates :math:`(x, y)`. It is one of the simplest example of the reaction-diffusion equation. In the literarure, this equation is called the Skellam equation. In 1951, he proposed the model of invasion of the muskrat in Europe. Muskrat (Ondatra zibethica) is a large rodent closely related to rats, mice, voles, hamster and lemmings. Muskrat is also known as a North American aquatic rodent. Muskrats possess wetlands, swamps, and zones close to the lakes, ponds, and streams. In specific parts of the world, muskrats are thought to be pests since they harm waterway dams and feast upon the yields. Individuals chased muskrats in the past just because of their fur and meat. In history, they had been the most trapped animals, for their fur having an economic value. It is interesting that only 5 muskrats released near Prague in 1905 was the source of rapid expansion year after year. Many millions of muskrats caused a big agriculture and ecological problems. The impact of muskrats on aquatic system is in general negative, especialli in the central Europe. It is a classic example what may happen when animals are transferred into new regions. .. figure:: http://upload.wikimedia.org/wikipedia/commons/thumb/3/35/Bisamratte-drawing.jpg/250px-Bisamratte-drawing.jpg :alt: image :figclass: align-center Bisamratte In this model, we consider only such cases when :math:`N(x, y, t) \ge 0`. Aditionally, we need: (1) the initial condition :math:`N(x, y, 0)= N_0(x, y)` (2) boundary conditions The above equation can be solved by the substitution .. math:: N(x, y, t)=e^{kt} c(x, y, t) where $c(x, y, t) is a new unknown function. If we insert it to the reaction-diffusion equation, we see that the new function :math:`c(x, y, t)` fulfils the equation .. math:: \frac{d c(x, y, t)}{dt} = D \left[ \frac{\partial^2 }{\partial x^2} + \frac{\partial^2 }{\partial y^2} \right] c(x, y, t) It is a standard diffusion equation in the 2-dimensional space and its solution is presented in the previous part. So, the function :math:`N(x, y, t)` has the form .. math:: N(x, y, t)=e^{kt} c(x, y, t) = \frac{N_0}{4\pi Dt} \; exp\left[kt -\frac{x^2+y^2}{4Dt}\right] where stała :math:`N_0=N(0)` is a concentration of individuals in population at the initial time :math:`t=0`. Because :math:`N(x, y, t)` is a number of individuals per unit of area, hence .. math:: \int_{-\infty}^{\;\infty} \;\int_{-\infty}^{\;\infty} N(x, y, t) \,dx \,dy = N(t) = N(0) e^{kt} Let us note that evolution of concentration depends on the coordinates in such a way .. math:: N(x, y, t)= N(r ,t) = \frac{N_0}{4\pi Dt} \; exp\left[kt -\frac{r^2}{4Dt}\right] where .. math:: r^2=x^2+y^2 is a distance from the origin of the coordinate system. Analysis of :math:`N(r, t)` can be performed with the help of Sage. Below we present graphical presentation. .. raw:: html
Failed to display Jupyter Widget of type sage_interactive
.
If you're reading this message in the Jupyter Notebook or JupyterLab Notebook, it may mean that the widgets JavaScript is still loading. If this message persists, it likely means that the widgets JavaScript library is either not installed or not enabled. See the Jupyter Widgets Documentation for setup instructions.
If you're reading this message in another frontend (for example, a static rendering on GitHub or NBViewer), it may mean that your frontend doesn't currently support widgets.
Numerical analysis of the Fisher-Kolmogorov equation: ----------------------------------------------------- .. math:: \frac{\partial n(x, t)}{\partial t} = r n(x, t) \;[1- n(x, t)] + D \frac{\partial^2 n(x, t)}{\partial x^2} For numerical solution of this equation, we take the simplest discrete verion of the differential equation using the definition of the derivative: .. math:: \frac{dg(t)}{dt} = B(g(t)) \quad \to \quad g^{i+1} = g^i + dt\; B(g^i) and for the Laplace operator: .. math:: \frac{d^2 f(x)}{dx^2} \quad \to \quad Lf^k = (dx)^2 \; [f^{k+1} - 2f^k + f^{k-1}] Then the discrete form of Fisher-Kolmogorov equation takes the form: .. math:: u^{i+1} = u^i + dt \left( r u^i (1-u^i) + D \frac{1}{h^2} Lu^i\right), We assume the periodic boundary conditions by an appropriate choice of the discrete version of the Laplacian. We discretize the space (here the line) and next we fix the time step. In order obtain solutions of the traveling wave one has to assume small value od the diffusion constant less than :math:`D\lt 0.001` in the vicinity of :math:`r=0`. The length of the system can be incorporated in :math:`D`, so one can take the length as 1. .. code:: ipython2 import numpy as np Dyf = 1.0 r = 1.0 l = 100.0 # dlugosc ukladu t_end = 50 # czas symulacje N = 100 # dyskretyzacja przestrzeni h = l/(N-1) dt = 0.2/(Dyf*(N-1)**2/l**2) # 0.2 z warunku CFL, krok nie moze byc wiekszy sps = int(1/dt) # liczba krokow na jednostke czasu Nsteps=sps*t_end # calkowita liczba krotkow print "sps=",sps,"dt=",dt one = np.identity(N) L=np.roll(one,-1)+np.roll(one,1)-2*one L[0,0]=-2. L[-1,-1]=-2. L[0,-1]=1. L[-1,0]=1. # warunek poczatkowy u = np.zeros(N) u[N/2-10:N/2+10]=.1 # small bump #u[:N/2]=1 # step Tlst=[] for i in range(Nsteps): if not i%sps: Tlst.append(list(u)) u = u + dt*(r*u*(1-u) + Dyf*(N-1)**2/l**2*L.dot(u)) @interact def _(ti=slider(range(len(Tlst)))): print r"tau=",dt*ti p = list_plot(Tlst[ti],plotjoined=True) p += list_plot(Tlst[-1],plotjoined=True,color='red',ymin=0,ymax=1.5) p += list_plot(Tlst[0],plotjoined=True,color='gray') p.show(figsize=(8,3)) .. parsed-literal:: sps= 4 dt= 0.204060810121416 .. raw:: htmlFailed to display Jupyter Widget of type sage_interactive
.
If you're reading this message in the Jupyter Notebook or JupyterLab Notebook, it may mean that the widgets JavaScript is still loading. If this message persists, it likely means that the widgets JavaScript library is either not installed or not enabled. See the Jupyter Widgets Documentation for setup instructions.
If you're reading this message in another frontend (for example, a static rendering on GitHub or NBViewer), it may mean that your frontend doesn't currently support widgets.
.. code:: ipython2 import numpy as np Dyf = 1.0 r = 1.0 l = 100.0 # dlugosc ukladu t_end = 100 # czas symulacje N = 200 # dyskretyzacja przestrzeni h = l/(N-1) dt = 0.052/(Dyf*(N-1)**2/l**2) # 0.2 z warunku CFL, krok nie moze byc wiekszy sps = int(1/dt) # liczba krokow na jednostke czasu Nsteps=sps*t_end # calkowita liczba krotkow print "sps=",sps,"dt=",dt one = np.identity(N) L=np.roll(one,-1)+np.roll(one,1)-2*one L[0,0]=1. L[-1,-1]=1. # warunek poczatkowy u = np.zeros(N) #u[:int(N/2]=1 # step for i in range(1,3): u[i] = 1.0 - i/3.0 def essential_boundary_conditions(u): u[0] = 1. u[-1] = 0.0 Tlst=[] essential_boundary_conditions(u) for i in range(Nsteps): if not i%sps: Tlst.append(list(u)) u = u + dt*(r*u*(1-u) + Dyf*(N-1)**2/l**2*L.dot(u)) essential_boundary_conditions(u) @interact def _(ti=slider(range(len(Tlst)))): print r"t=",dt*ti p = list_plot(Tlst[ti],plotjoined=True) p += list_plot(Tlst[-1],plotjoined=True,color='red',ymin=0,ymax=1.5) p += list_plot(Tlst[0],plotjoined=True,color='gray') p.show(figsize=(8,3)) .. parsed-literal:: sps= 76 dt= 0.0131309815408702 .. raw:: htmlFailed to display Jupyter Widget of type sage_interactive
.
If you're reading this message in the Jupyter Notebook or JupyterLab Notebook, it may mean that the widgets JavaScript is still loading. If this message persists, it likely means that the widgets JavaScript library is either not installed or not enabled. See the Jupyter Widgets Documentation for setup instructions.
If you're reading this message in another frontend (for example, a static rendering on GitHub or NBViewer), it may mean that your frontend doesn't currently support widgets.
Speed of the wave front ~~~~~~~~~~~~~~~~~~~~~~~ .. code:: ipython2 pos_lst = [] for T_ in Tlst: for (i,a),b in zip(enumerate(T_),T_[1:]): if a>=0.5 and b<=0.5: pos_lst.append( i+(a-0.5)/(a-b) ) list_plot( [l/(N-1)*(b-a)/(sps*dt) for a,b in zip(pos_lst,pos_lst[1:])] , figsize=(7,3),gridlines=[[],[2]],ymax=2) .. image:: output_15_0.png