24. The Solow-Swan Growth Model#

In this lecture we review a famous model due to Robert Solow (1925–2023) and Trevor Swan (1918–1989).

The model is used to study growth over the long run.

Although the model is simple, it contains some interesting lessons.

We will use the following imports.

import matplotlib.pyplot as plt
import numpy as np

24.1. The model#

In a Solow–Swan economy, agents save a fixed fraction of their current incomes.

Savings sustain or increase the stock of capital.

Capital is combined with labor to produce output, which in turn is paid out to workers and owners of capital.

To keep things simple, we ignore population and productivity growth.

For each integer t0, output Yt in period t is given by Yt=F(Kt,Lt), where Kt is capital, Lt is labor and F is an aggregate production function.

The function F is assumed to be nonnegative and homogeneous of degree one, meaning that

F(λK,λL)=λF(K,L)for all λ0

Production functions with this property include

  • the Cobb-Douglas function F(K,L)=AKαL1α with 0α1.

  • the CES function F(K,L)={aKρ+bLρ}1/ρ with a,b,ρ>0.

Here, α is the output elasticity of capital and ρ is a parameter that determines the elasticity of substitution between capital and labor.

We assume a closed economy, so aggregate domestic investment equals aggregate domestic saving.

The saving rate is a constant s satisfying 0s1, so that aggregate investment and saving both equal sYt.

Capital depreciates: without replenishing through investment, one unit of capital today becomes 1δ units tomorrow.

Thus,

Kt+1=sF(Kt,Lt)+(1δ)Kt

Without population growth, Lt equals some constant L.

Setting kt:=Kt/L and using homogeneity of degree one now yields

kt+1=sF(Kt,L)L+(1δ)KtL=sF(Kt,L)L+(1δ)kt=sF(kt,1)+(1δ)kt

With f(k):=F(k,1), the final expression for capital dynamics is

(24.1)#kt+1=g(kt) where g(k):=sf(k)+(1δ)k

Our aim is to learn about the evolution of kt over time, given an exogenous initial capital stock k0.

24.2. A graphical perspective#

To understand the dynamics of the sequence (kt)t0 we use a 45-degree diagram.

To do so, we first need to specify the functional form for f and assign values to the parameters.

We choose the Cobb–Douglas specification f(k)=Akα and set A=2.0, α=0.3, s=0.3 and δ=0.4.

The function g from (24.1) is then plotted, along with the 45-degree line.

Let’s define the constants.

A, s, alpha, delta = 2, 0.3, 0.3, 0.4
x0 = 0.25
xmin, xmax = 0, 3

Now, we define the function g.

def g(A, s, alpha, delta, k):
    return A * s * k**alpha + (1 - delta) * k

Let’s plot the 45-degree diagram of g.

def plot45(kstar=None):
    xgrid = np.linspace(xmin, xmax, 12000)

    fig, ax = plt.subplots()

    ax.set_xlim(xmin, xmax)

    g_values = g(A, s, alpha, delta, xgrid)

    ymin, ymax = np.min(g_values), np.max(g_values)
    ax.set_ylim(ymin, ymax)

    lb = r'$g(k) = sAk^{\alpha} + (1 - \delta)k$'
    ax.plot(xgrid, g_values,  lw=2, alpha=0.6, label=lb)
    ax.plot(xgrid, xgrid, 'k-', lw=1, alpha=0.7, label='$45^{\circ}$')

    if kstar:
        fps = (kstar,)

        ax.plot(fps, fps, 'go', ms=10, alpha=0.6)

        ax.annotate(r'$k^* = (sA / \delta)^{(1/(1-\alpha))}$',
                 xy=(kstar, kstar),
                 xycoords='data',
                 xytext=(-40, -60),
                 textcoords='offset points',
                 fontsize=14,
                 arrowprops=dict(arrowstyle="->"))

    ax.legend(loc='upper left', frameon=False, fontsize=12)

    ax.set_xticks((0, 1, 2, 3))
    ax.set_yticks((0, 1, 2, 3))

    ax.set_xlabel('$k_t$', fontsize=12)
    ax.set_ylabel('$k_{t+1}$', fontsize=12)

    plt.show()
<>:15: SyntaxWarning: invalid escape sequence '\c'
<>:15: SyntaxWarning: invalid escape sequence '\c'
/tmp/ipykernel_6933/3107437576.py:15: SyntaxWarning: invalid escape sequence '\c'
  ax.plot(xgrid, xgrid, 'k-', lw=1, alpha=0.7, label='$45^{\circ}$')
plot45()
_images/eb2a0ba00ca8d7a87326646dfcc72fd65d591f85b30fd9993efc3e91ac1e034b.png

Suppose, at some kt, the value g(kt) lies strictly above the 45-degree line.

Then we have kt+1=g(kt)>kt and capital per worker rises.

If g(kt)<kt then capital per worker falls.

If g(kt)=kt, then we are at a steady state and kt remains constant.

(A steady state of the model is a fixed point of the mapping g.)

From the shape of the function g in the figure, we see that there is a unique steady state in (0,).

It solves k=sAkα+(1δ)k and hence is given by

(24.2)#k:=(sAδ)1/(1α)

If initial capital is below k, then capital increases over time.

If initial capital is above this level, then the reverse is true.

Let’s plot the 45-degree diagram to show the k in the plot.

kstar = ((s * A) / delta)**(1/(1 - alpha))
plot45(kstar)
_images/fc30c38bacfd4aa9f3343cbf24614d332f18cc531cca19de891e9ba0d220de54.png

From our graphical analysis, it appears that (kt) converges to k, regardless of initial capital k0.

This is a form of global stability.

The next figure shows three time paths for capital, from three distinct initial conditions, under the parameterization listed above.

At this parameterization, k1.78.

Let’s define the constants and three distinct initial conditions

A, s, alpha, delta = 2, 0.3, 0.3, 0.4
x0 = np.array([.25, 1.25, 3.25])

ts_length = 20
xmin, xmax = 0, ts_length
ymin, ymax = 0, 3.5
def simulate_ts(x0_values, ts_length):

    k_star = (s * A / delta)**(1/(1-alpha))
    fig, ax = plt.subplots(figsize=[11, 5])
    ax.set_xlim(xmin, xmax)
    ax.set_ylim(ymin, ymax)

    ts = np.zeros(ts_length)

    # simulate and plot time series
    for x_init in x0_values:
        ts[0] = x_init
        for t in range(1, ts_length):
            ts[t] = g(A, s, alpha, delta, ts[t-1])
        ax.plot(np.arange(ts_length), ts, '-o', ms=4, alpha=0.6,
                label=r'$k_0=%g$' %x_init)
    ax.plot(np.arange(ts_length), np.full(ts_length,k_star),
            alpha=0.6, color='red', label=r'$k^*$')
    ax.legend(fontsize=10)

    ax.set_xlabel(r'$t$', fontsize=14)
    ax.set_ylabel(r'$k_t$', fontsize=14)

    plt.show()
simulate_ts(x0, ts_length)
_images/cbc9f4786772249302319ae8e7a37935cad9515e3ec072122ccfb57cb38dadfe.png

As expected, the time paths in the figure all converge to k.

24.3. Growth in continuous time#

In this section, we investigate a continuous time version of the Solow–Swan growth model.

We will see how the smoothing provided by continuous time can simplify our analysis.

Recall that the discrete time dynamics for capital are given by kt+1=sf(kt)+(1δ)kt.

A simple rearrangement gives the rate of change per unit of time:

Δkt=sf(kt)δktwhereΔkt:=kt+1kt

Taking the time step to zero gives the continuous time limit

(24.3)#kt=sf(kt)δktwithkt:=ddtkt

Our aim is to learn about the evolution of kt over time, given an initial stock k0.

A steady state for (24.3) is a value k at which capital is unchanging, meaning kt=0 or, equivalently, sf(k)=δk.

We assume f(k)=Akα, so k solves sAkα=δk.

The solution is the same as the discrete time case—see (24.2).

The dynamics are represented in the next figure, maintaining the parameterization we used above.

Writing kt=g(kt) with g(k)=sAkαδk, values of k with g(k)>0 imply kt>0, so capital is increasing.

When g(k)<0, the opposite occurs. Once again, high marginal returns to savings at low levels of capital combined with low rates of return at high levels of capital combine to yield global stability.

To see this in a figure, let’s define the constants

A, s, alpha, delta = 2, 0.3, 0.3, 0.4

Next we define the function g for growth in continuous time

def g_con(A, s, alpha, delta, k):
    return A * s * k**alpha - delta * k
def plot_gcon(kstar=None):

    k_grid = np.linspace(0, 2.8, 10000)

    fig, ax = plt.subplots(figsize=[11, 5])
    ax.plot(k_grid, g_con(A, s, alpha, delta, k_grid), label='$g(k)$')
    ax.plot(k_grid, 0 * k_grid, label="$k'=0$")

    if kstar:
        fps = (kstar,)

        ax.plot(fps, 0, 'go', ms=10, alpha=0.6)


        ax.annotate(r'$k^* = (sA / \delta)^{(1/(1-\alpha))}$',
                 xy=(kstar, 0),
                 xycoords='data',
                 xytext=(0, 60),
                 textcoords='offset points',
                 fontsize=12,
                 arrowprops=dict(arrowstyle="->"))

    ax.legend(loc='lower left', fontsize=12)

    ax.set_xlabel("$k$",fontsize=10)
    ax.set_ylabel("$k'$", fontsize=10)

    ax.set_xticks((0, 1, 2, 3))
    ax.set_yticks((-0.3, 0, 0.3))

    plt.show()
kstar = ((s * A) / delta)**(1/(1 - alpha))
plot_gcon(kstar)
_images/53d167007c17b9383b6f28b82560e13a18f37b38c767901492506b6912171ba3.png

This shows global stability heuristically for a fixed parameterization, but how would we show the same thing formally for a continuum of plausible parameters?

In the discrete time case, a neat expression for kt is hard to obtain.

In continuous time the process is easier: we can obtain a relatively simple expression for kt that specifies the entire path.

The first step is to set xt:=kt1α, so that xt=(1α)ktαkt.

Substituting into kt=sAktαδkt leads to the linear differential equation

(24.4)#xt=(1α)(sAδxt)

This equation, which is a linear ordinary differential equation, has the solution

xt=(k01αsAδ)eδ(1α)t+sAδ

(You can confirm that this function xt satisfies (24.4) by differentiating it with respect to t.)

Converting back to kt yields

(24.5)#kt=[(k01αsAδ)eδ(1α)t+sAδ]1/(1α)

Since δ>0 and α(0,1), we see immediately that ktk as t independent of k0.

Thus, global stability holds.

24.4. Exercises#

Exercise 24.1

Plot per capita consumption c at the steady state, as a function of the savings rate s, where 0s1.

Use the Cobb–Douglas specification f(k)=Akα.

Set A=2.0,α=0.3, and δ=0.5

Also, find the approximate value of s that maximizes the c(s) and show it in the plot.

Exercise 24.2

Stochastic Productivity

To bring the Solow–Swan model closer to data, we need to think about handling random fluctuations in aggregate quantities.

Among other things, this will eliminate the unrealistic prediction that per-capita output yt=Aktα converges to a constant y:=A(k)α.

We shift to discrete time for the following discussion.

One approach is to replace constant productivity with some stochastic sequence (At)t1.

Dynamics are now

(24.6)#kt+1=sAt+1f(kt)+(1δ)kt

We suppose f is Cobb–Douglas and (At) is IID and lognormal.

Now the long run convergence obtained in the deterministic case breaks down, since the system is hit with new shocks at each point in time.

Consider A=2.0,s=0.6,α=0.3, and δ=0.5

Generate and plot the time series kt.