Search code examples
juliapderunge-kutta

Julia challenge - FitzHugh–Nagumo model PDE Runge-Kutta solver


I am newbie in Julia programming language, so I don't know much of how to optimize a code. I have heard that Julia should be faster in comparison to Python, but I've written a simple Julia code for solving the FitzHugh–Nagumo model , and it doesn't seems to be faster than Python.

The FitzHugh–Nagumo model equations are:

function FHN_equation(u,v,a0,a1,d,eps,dx)
  u_t = u - u.^3 - v + laplacian(u,dx)
  v_t = eps.*(u - a1 * v - a0) + d*laplacian(v,dx)
  return u_t, v_t
end

where u and v are the variables, which are 2D fields (that is, 2 dimensional arrays), and a0,a1,d,eps are the model's parameters. Both parameters and the variables are of type Float. dx is the parameter that control the separation between grid point, for the use of the laplacian function, which is an implementation of finite differences with periodic boundary conditions.

If one of you expert Julia coders can give me a hint of how to do things better in Julia I will be happy to hear.

The Runge-Kutte step function is:

function uv_rk4_step(Vs,Ps, dt)
  u = Vs.u
  v = Vs.v
  a0=Ps.a0
  a1=Ps.a1
  d=Ps.d
  eps=Ps.eps
  dx=Ps.dx
  du_k1, dv_k1 =    FHN_equation(u,v,a0,a1,d,eps,dx)
  u_k1 = dt*du_k1י
  v_k1 = dt*dv_k1
  du_k2, dv_k2 =    FHN_equation((u+(1/2)*u_k1),(v+(1/2)*v_k1),a0,a1,d,eps,dx)
  u_k2 = dt*du_k2
  v_k2 = dt*dv_k2
  du_k3, dv_k3 =    FHN_equation((u+(1/2)*u_k2),(v+(1/2)*v_k2),a0,a1,d,eps,dx)
  u_k3 = dt*du_k3
  v_k3 = dt*dv_k3
  du_k4, dv_k4 =    FHN_equation((u+u_k3),(v+v_k3),a0,a1,d,eps,dx)
  u_k4 = dt*du_k4
  v_k4 = dt*dv_k4
  u_next    =   u+(1/6)*u_k1+(1/3)*u_k2+(1/3)*u_k3+(1/6)*u_k4
  v_next    =   v+(1/6)*v_k1+(1/3)*v_k2+(1/3)*v_k3+(1/6)*v_k4
  return u_next, v_next
end

And I've used imshow() from PyPlot package to plot the u field.


Solution

  • This is not a complete answer, but a taste of an optimization attempt on the laplacian function. The original laplacian on a 10x10 matrix gave me the @time:

    0.000038 seconds (51 allocations: 12.531 KB)
    

    While this version:

    function laplacian2(a,dx)
              # Computes Laplacian of a matrix
              # Usage: al=laplacian(a,dx)
              # where  dx is the grid interval
              ns=size(a,1)
              ns != size(a,2) && error("Input matrix must be square")
              aa=zeros(ns+2,ns+2)
    
              for i=1:ns
                  aa[i+1,1]=a[i,end]
                  aa[i+1,end]=a[i,1]
                  aa[1,i+1]=a[end,i]
                  aa[end,i+1]=a[1,i]
              end
              for i=1:ns,j=1:ns
                  aa[i+1,j+1]=a[i,j]
              end
              lap = Array{eltype(a),2}(ns,ns)
              scale = inv(dx*dx)
              for i=1:ns,j=1:ns
                  lap[i,j]=(aa[i,j+1]+aa[i+2,j+1]+aa[i+1,j]+aa[i+1,j+2]-4*aa[i+1,j+1])*scale
              end
              return lap
    end
    

    Gives @time:

    0.000010 seconds (6 allocations: 2.250 KB)
    

    Notice the reduction in allocations. Extra allocations usually indicate the potential for optimization.