←to practical programming

Homework "ODE"

Objective

Implement an embedded Runge-Kutta stepper with error estimate, and an adaptive-step-size driver for solving Ordinary Differential Equation Initial Value Problems (ODE IVP).

Tasks

  1. (6 points) Embedded rule Runge-Kutta ODE integrator
    1. Implement an embedded Runge-Kutta stepper rkstepXY (where XY describes the order of the imbedded method used, for example XY=12—"one-two"—for the midpoint-Euler method) of your choice, which advances the solution to the equation

      dy/dx = f(x,y)
      (where y and f are vectors) by a given step h, and estimates the error.

      Something like this (Runge-Kutta Euler/Midpoint 12-method, you might want to do better),

      public static (vector,vector) rkstep12(
      	Func<double,vector,vector> f,/* the f from dy/dx=f(x,y) */
      	double x,                    /* the current value of the variable */
      	vector y,                    /* the current value y(x) of the sought function */
      	double h                     /* the step to be taken */
      	)
      {
      	vector k0 = f(x,y);              /* embedded lower order formula (Euler) */
      	vector k1 = f(x+h/2,y+k0*(h/2)); /* higher order formula (midpoint) */
      	vector yh = y+k1*h;              /* y(x+h) estimate */
      	vector δy = (k1-k0)*h;           /* error estimate */
      	return (yh,δy);
      }
      
    2. Implement an adaptive-step-size driver routine wchich advances the solution from initial-point a to final-point b (by calling your rkstepXY routine with appropriate step-sizes) keeping the specified relative, eps, and absolute, acc, precision. You driver should record the solution in two generic lists, "genlist<double> x" and "genlist<vector> y" and then return the two lists.

      The interface could be like this,

      public static (genlist<double>,genlist<vector>) driver(
      	Func<double,vector,vector> F,/* the f from dy/dx=f(x,y) */
      	(double,double) interval,    /* (initial-point,final-point) */
      	vector yinit,                /* y(initial-point) */
      	double h=0.125,              /* initial step-size */
      	double acc=0.01,             /* absolute accuracy goal */
      	double eps=0.01              /* relative accuracy goal */
      ){
      var (a,b)=interval; double x=a; vector y=yinit.copy();
      var xlist=new genlist<double>(); xlist.add(x);
      var ylist=new genlist<vector>(); ylist.add(y);
      do{
      	if(x>=b) return (xlist,ylist); /* job done */
      	if(x+h>b) h=b-x;               /* last step should end at b */
      	var (yh,δy) = rkstep12(F,x,y,h);
      	double tol = (acc+eps*yh.norm()) * Sqrt(h/(b-a));
      	double err = δy.norm();
      	if(err<=tol){ // accept step
      		x+=h; y=yh;
      		xlist.add(x);
      		ylist.add(y);
      		}
      	if(err>0) h *= Min( Pow(tol/err,0.25)*0.95 , 2); // readjust stepsize
      	else h*=2;
      	}while(true);
      }//driver
      
    3. Debug your routines by solving some interesting systems of ordinary differential equations, for example u''=-u.

    4. Reproduce (with your routines) the example from the scipy.integrate.odeint manual (oscillator with friction) and/or the example from the scipy.integrate.solve_ivp manual (Lotka-Volterra system).

  2. (3 points) Relativistic precession of planetary orbit

    Consider the equation (in certain units) of equatorial motion of a planet around a star in General Relativity,

    u''(φ) + u(φ) = 1 + εu(φ)2 ,

    where u(φ) ≡ 1/r(φ) , r is the (reduced-circumference) radial coordinate, φ is the azimuthal angle, ε is the relativistic correction (on the order of the star's Schwarzschild radius divided by the radius of the planet's orbit), and primes denote the derivative with respect to φ.

    1. Integrate (for several rotations) this equation with ε=0 and initial conditions u(0)=1, u'(0)=0 : this should give a Newtonian circular motion.

    2. Integrate (for several rotations) this equation with ε=0 and initial conditions u(0)=1, u'(0)≈-0.5 : this should give a Newtonian elliptical motion. Hint: u'(0) shouldn't bee too large or you will lose your planet.

    3. Integrate (for several rotations) this equation with ε≈0.01 and initial conditions u(0)=1, u'(0)≈-0.5 : this should illustrate the relativistic precession of a planetary orbit.

    Hints:

  3. (1 points) Test of the order of the method; Alternative interface; Newtonian gravitational three-body problem
    1. Implement an imbedded stepper of order 23. Test whether it intergates the equation y''=2x exactly: solve this equation numerically, and check that the driver keeps doubling the step-size (as the error from the stepper should be close to zero) still producing the exact result.

    2. Implement a quadratic spline interpolation routine that interpolates tables, {xi,yi}, of vector-valued functions y=f(x). Something like this,

      public static Func<double,vector> make_qspline(genlist<double> x,genlist<vector> y)
      {
      	/* calculate bi and ci of the quadratic spline */
      	Func<double,vector> qspline = delegate(double z){
      		int i=binsearch(x,z);
      		return y[i]+(z-x[i])*(b[i] + c[i]*(z-x[i]));;
      	};
      	return qspline;
      }
      
    3. Change the interface of your driver such that it returns not the calculated table but the quadratic spline of the table,
      public static Func<double,vector> make_ode_ivp_qspline
      (Func<double,vector,vector> F,(double,double)interval,vector y,
      double acc=0.01,double eps=0.01,double hstart=0.01 )
      {
      	var (xlist,ylist) = driver(F,interval,y,acc,eps,hstart);
      	return make_qspline(xlist,ylist);
      }
      
    4. The dynamics of the Newtonian gravitational three-body problem is generally chaotic. However, there apparently exists a remarkable stable planar periodic solution in the shape of figure-8 [Wikipedia: Special-case solutions; Alain Chenciner, Richard Montgomeryi, arXiv:math/0011268].

      Using your numerical ODE integrator reproduce this solution.

      Hints:

      • The Newtonian equations of motion for several gravitating bodies with masses mi is given as
        midvi/dt = ∑j≠i ( Gmimj /|rj-ri|2 ) ( rj-ri /|rj-ri| ).
      • You can set m1=m2=m3=1, G=1 (that amounts to a particular choice of units). The equation of motion in these units should looke like
        vi' = ∑j≠i rj-ri /|rj-ri|3 .
      • Now you can combine the three coordinate-vectors ri ad the three velocities ri' (they are all 2D because we have got a planar motion) into one 12-component column,
        z = {x'1, y'1, x'2, y'2, x'3, y'3, x1, y1, x2, y2, x3, y3}
        and rewrite the above equations through the z-components: that would be the equation to solve with your ODE solver. The initial conditions can be found in the Wikipedia article.