Some Numerical Examples#

Now that we’ve seen a C++ program at its most basic, let’s look at some more useful examples. Again we’re going to start from code you’ve probably already seen working in Python and convert it into C++ instead.

A prime calculator#

Let’s start off with some Python code to list the prime numbers less than 1000.

primes.py

    # A simple primes calculator
    for i in range(2, 1000):
        j = 2
        flag = True
        while j**2<=i:
            if i%j == 0:
                flag = False
                break
            j += 1
        if flag:
            print(i)

Now let’s convert it into a C++ version

primes.cpp

    // A simple primes calculator
    #include <iostream>

    int main() {
        for (int i=2; i<1000; i++) {
            int j = 2;
            bool flag = true;
            while (j*j<=i) {
                if (i%j==0) {
                    flag = false;
                    break;
                }
                j++;
            }
            if (flag) {
                std::cout << i << std::endl;
            }
        }
        return 1;
    }

Once again we need to include the iostream header (to be able to write to the screen) and to place our code inside a main function so that it runs when our program is started. The other big difference is that our variables need to be declared to have particular datatypes (e.g. int j) at or before their first point of use. This is important, and the code will not compile without it, since the C++ compiler uses the type information to work out which versions of functions to call and to make sure that users have not made mistakes.

Note that we’ve also included a comment in our code. In C/C++ one-line comments start with //. We can also write an entire comment block:

/* This text will be ignored by a compiler.
And so will this!
To end the comment we need a */

int main(){
    a = 1;
    a--;
    return a;
}

The forward Euler method#

Let’s now look at a more complicated example, the forward Euler method for solving a simple ODE. We’ll start with the Python code:

forward_euler.py

    # A simple forward Euler solver for the ODE dy/dt = -y
    import numpy as np
    import matplotlib.pyplot as plt

    def f(y):
        return -y

    y0 = 1
    t0 = 0
    t1 = 10
    h = 0.1

    t = np.arange(t0, t1+h, h)
    y = np.zeros_like(t)
    y[0] = y0

    for i in range(1, len(t)):
        y[i] = y[i-1] + h*f(y[i-1])

    plt.plot(t, y)
    plt.show()

Now let’s convert it into a C++ version

#include <iostream>

double f(double y) {
    return -y;
}

int main(void){

    const int nsteps = 100;
    double y0 = 1;
    double t0 = 0;
    double t1 = 10;
    double h = t1/nsteps;

    double t[nsteps+1];
    double y[nsteps+1];

    t[0] = t0;
    y[0] = y0;

    for (int i=1; i<=nsteps; i++) {
        t[i] = t[i-1] + h;
        y[i] = y[i-1] + h*f(y[i-1]);
    }

    for (int i=0; i<n; i++) {
        std::cout << t[i] << " " << y[i] << std::endl;
    }

    return 0;
}

Note that unlike in Python, plotting in C++ is hard work, so we’re just printing the values to screen instead.

Quadrature#

Now we’ll implement two quadrature methods, the midpoint and the trapezium rules. We’ll start with the Python code again (see the Computational Mathematics course notes for the annotated versions):


import numpy as np

def midpoint_rule(a, b, func, number_intervals=10):
    interval_size = (b - a)/number_intervals

    I_M = 0.0
    mid = a + (interval_size/2.0)

    while (mid < b):
        I_M += interval_size * func(mid)
        mid += interval_size
    return I_M

def trapezium_rule(a, b, func, number_intervals=10):
    interval_size = (b - a)/number_intervals

    I_T = 0.0
    x = a + interval_size

    while (x < b):
        I_T += interval_size * (func(x) + func(x - interval_size))/2.0
        x += interval_size
    return I_T

def f(x):
    return np.sin(x)

print(midpoint_rule(0, np.pi, f))
print(trapezium_rule(0, np.pi, f))

Now let’s convert it into a C++ version:

#include <iostream> // for std::cout
#include <cmath> // for std::sin

double midpoint_rule(double a, double b, double (*func)(double), int number_intervals=10) {
    double interval_size = (b - a)/number_intervals;

    double I_M = 0.0;
    double mid = a + (interval_size/2.0);

    while (mid < b) {
        I_M += interval_size * func(mid);
        mid += interval_size;
    }
    return I_M;
}

double trapezium_rule(double a, double b, double (*func)(double), int number_intervals=10) {

    double interval_size = (b - a)/number_intervals;

    double I_T = 0.0;
    double x = a + interval_size;

    while (x < b) {
        I_T += interval_size * (func(x) + func(x - interval_size))/2.0;
        x += interval_size;
    }
    return I_T;
}

double f(double x) {
    return std::sin(x);
}

int main(void) {
    std::cout << midpoint_rule(0, M_PI, f) << std::endl;
    std::cout << trapezium_rule(0, M_PI, f) << std::endl;
    return 0;
}

Note that we’ve had to include the cmath header to get access to the sin function. We’ve also had to declare the function f as taking a double as input and returning a double, and then pass it in as an argument to the quadrature functions. This is because C++ doesn’t have a single unique built-in sin function, but instead has a sin function for each of the floating point types (e.g. float, double, long double) in its cmath library, and the compiler needs to know which one we want to use.

The declaration and use of function pointers in the C++ versions of the midpoint_rule and trapezium_rule is more complicated than the Python version, but it’s not too impossible to remember once you get used to it. We’ll talk more about this in a later section.

Exercises#

  1. Try to modify the C++ primes code to calulate the first 100 prime numbers instead.

  2. Try to write a C++ version of the backwards Euler method for the same ODE.

  3. Try to write a C++ version of the Simpson’s rule quadrature method, and include it in the quadrature example.

Summary#

We’ve now seen C++ used to solve some simple numerical problems of the sort which you’re already used to solving with Python. In the next section we’ll look at some ways of modifying the behaviour of our code at runtime based on user input.