Understanding the Predictably Unpredictable¶
Have you ever wondered why weather forecasts become unreliable beyond a few days? Or why your heart beats with slight irregularities even when you're perfectly healthy? These phenomena share a fascinating common thread: chaos theory, a branch of mathematics that explains how tiny changes can lead to dramatically different outcomes.
Chaos theory isn't just an academic curiosity - it's everywhere around us. From the turbulent flow of water from your faucet to the fluctuations in financial markets, from the patterns of animal populations to the spread of epidemics, chaos theory helps us understand complex systems that at first glance appear random but actually follow hidden patterns.
In this interactive tutorial, we'll use the power of Pluto notebooks and Julia programming to explore chaos theory hands-on. You'll discover how simple mathematical rules can generate surprisingly complex behavior, and learn why some systems are fundamentally unpredictable despite being completely deterministic.
Learning Objectives¶
By the end of this tutorial, you will be able to:
- Set up and use Pluto notebooks for interactive mathematical exploration
- Understand how linear systems behave differently from chaotic ones
- Visualize and analyze the behavior of the logistic function
- Interpret phase portraits and autocorrelation plots
- Recognize sensitive dependence on initial conditions, a key feature of chaos
- Apply these concepts to understand real-world chaotic systems
No advanced mathematics or programming experience is required - just bring your curiosity about how complexity emerges from simplicity. Let's dive in and make sense of chaos!
Getting Started with Pluto¶
Pluto is a notebook environment for Julia. Install Julia and start it to get a command window. The first step is to add some packages we'll use for this project. Type "]" at the prompt (without the quotes) to enter the package manager. The packages you'll need are:
add Pluto
add PlutoUI
add Plots
add StatsBase
It may take a few minutes as Julia checks and downloads all necessary dependencies. When everything has finished, type "Ctrl-C" to exit the package manager. In the Julia command window, type "Using Pluto" followed by "Pluto.run()":
A new tab will open in your browser:
Download EasyChaos.jl from Github and then enter the path in Pluto below "Open from file:". The notebook should look very much like the remainder of this post.
Easy Chaos!¶
Pluto is a programming environment for Julia, designed to be interactive and helpful. Changes made anywhere in the notebook affect the entire notebook. Pluto is reactive so you don't need to recalculate cells.
This is useful in studying chaos because small changes made to a function can have big effects. You can define a function, run it, and immediately see the effects.
A Linear Equation¶
The equation of a line is y = m x + b y = mx + b y=mx+b where m m m is the slope and b b b is the y − y- y−intercept. The slope is how much y y y changes for each unit change in x x x. For any starting point x x x if you move one step to the right to x + 1 x + 1 x+1, the new y − y- y−value becomes m ( x + 1 ) + b = m x + m + b m(x+1) + b = mx + m + b m(x+1)+b=mx+m+b. Subtract y = m x + b y = mx + b y=mx+b and all that's left is m m m.
A step by 1 1 1 in the x − x- x−direction gives a change by m m m in the y − y- y−direction, which is the slope of the line.
If x = 0 x=0 x=0, then y = m × 0 + b = b y = m \times 0 + b = b y=m×0+b=b so the y − y- y−intercept is b b b. Since two points define a line, then start at b b b on the y − y- y−axis, move to x = 1 x=1 x=1 and add m m m (or subtract m m m if m m m is negative) to get a second point. Draw your line through the two points.
The line can also be defined as a function,
f ( x ) = m x + b . f(x) = mx + b. f(x)=mx+b.Let's start by plotting some lines.
We need a way to change the values of m m m and b b b. First, we'll make a slider for the slope m m m and set the initial value to m = 1 m=1 m=1.
Next, we'll make a slider for b b b and set it to b = 0 b=0 b=0. Both can be varied between − 5 -5 −5 and 5 5 5.
Notice that the origin of the plot is in the center. Move the sliders for m m m and b b b to get a new plot. Interesting, but not chaos.
A Little Chaos¶
Chaos is doing nearly the same thing over and over and expecting wildly different results. Let's start with the linear equation above, with m = 1 m = 1 m=1 and b = − 1 b = -1 b=−1.
y = f ( x ) = x − 1. y = f(x) = x - 1. y=f(x)=x−1.Starting at x = 1 x = 1 x=1, the output of the function is y = 0 y=0 y=0. Using this new value as the input, f ( 0 ) = − 1 f(0) = -1 f(0)=−1. If you do this several times you get the uninteresting sequence
x = { 1 , 0 , − 1 , − 2 , − 3 , … } . x = \{1,0,-1,-2,-3, \ldots \}. x={1,0,−1,−2,−3,…}.Instead of calculating x 1 = m x 0 + b x_1 = mx_0 + b x1=mx0+b and then x 2 = m x 1 + b x_2 = mx_1 + b x2=mx1+b and so on, we can do this in a function iter_f with inputs m , b , x 0 , n m,b,x_0,n m,b,x0,n where x 0 x_0 x0 is the starting point, and n n n is the number of iterations.
We can try this function with the equation y = x − 1 y = x - 1 y=x−1 starting at x 0 = 0.4961 x_0 = 0.4961 x0=0.4961 and x 1 = 0.4962 x_1 = 0.4962 x1=0.4962 for n = 10 n = 10 n=10 iterations.
Taking the difference between the outputs gives an uninteresting answer:
Now, let's change the equation slightly to f ( x ) = 2 x − 1 f(x) = 2x - 1 f(x)=2x−1, and use the same starting points as before.
The difference between these sequences is:
Multiply the differences by 10000 10000 10000 to make the changes more obvious:
This is a little bit surprising, but notice that each number is twice the previous number. It looks like the sequence { 2 0 , 2 1 , 2 2 , … 2 9 } \{ 2^0, 2^1, 2^2, \dots 2^9 \} {20,21,22,…29} and could be written as 2 [ 0 : 9 ] 2^{[0:9]} 2[0:9]:
Let's write out several iterations of y = m x + b y = mx + b y=mx+b in terms of the starting point x 0 x_0 x0.
x 1 = m x 0 + b x 2 = m x 1 + b = m ( m x 0 + b ) + b = m 2 x 0 + m b + b = m 2 x 0 + ( m + 1 ) b x 3 = m x 2 + b = m ( m 2 x 0 + ( m + 1 ) b ) + b = m 3 x 0 + ( m 2 + m + 1 ) b x 4 = m x 3 + b = m ( m 3 x 0 + ( m 2 + m + 1 ) b ) + b = m 4 x 0 + ( m 3 + m 2 + m + 1 ) b \begin{aligned} x_1 &= m x_0 + b \\ x_2 &= m x_1 + b = m (m x_0 + b) + b = m^2 x_0 + m b + b = m^2 x_0 + (m+1) b \\ x_3 &= m x_2 + b = m (m^2 x_0 + (m+1) b) + b = m^3 x_0 + (m^2 + m + 1) b \\ x_4 &= m x_3 + b = m (m^3 x_0 + (m^2 + m +1) b) + b = m^4 x_0 + (m^3 + m^2 + m + 1) b \\ \end{aligned} x1x2x3x4=mx0+b=mx1+b=m(mx0+b)+b=m2x0+mb+b=m2x0+(m+1)b=mx2+b=m(m2x0+(m+1)b)+b=m3x0+(m2+m+1)b=mx3+b=m(m3x0+(m2+m+1)b)+b=m4x0+(m3+m2+m+1)bThe n t h n^{th} nth iterate in terms of x 0 x_0 x0 is
x n = m n x 0 + ( m n − 1 + m n − 2 + ⋯ + 1 ) b . x_n = m^n x_0 + (m^{n-1} + m^{n-2} + \cdots + 1) b. xn=mnx0+(mn−1+mn−2+⋯+1)b.What happens if we make a small change in x 0 x_0 x0? If we change x 0 x_0 x0 by a small amount, ϵ \epsilon ϵ, then
x ~ n = m n ( x 0 + ϵ ) + ( m n − 1 + m n − 2 + ⋯ + 1 ) b \tilde{x}_n = m^n (x_0 + \epsilon) + (m^{n-1} + m^{n-2} + \cdots + 1) b x~n=mn(x0+ϵ)+(mn−1+mn−2+⋯+1)band the difference between the two results after n n n iterations is (the term with b b b is the same for both)
x ~ n − x n = m n ( x 0 + ϵ ) − m n x 0 = m n ϵ . \tilde{x}_n - x_n = m^n (x_0 + \epsilon) - m^n x_0 = m^n \epsilon. x~n−xn=mn(x0+ϵ)−mnx0=mnϵ.So long as ∣ m ∣ > 1 |m| > 1 ∣m∣>1 then we can make m n ϵ m^n \epsilon mnϵ as big as we like by iterating enough times, no matter how small ϵ \epsilon ϵ is. A small change to the initial value of x 0 x_0 x0 produces as big a change as you like in the final value, x n x_n xn.
There you have it. Chaos!
Moar Chaos¶
With a slightly more complicated equation, chaos becomes even more interesting. Instead of iterating a linear equation, let's use the logistic function (see The Growing Gap for an application of logistic functions)
F ( x ) = c x ( 1 − x ) . F(x) = cx(1-x). F(x)=cx(1−x).For c = 4 c = 4 c=4, this is an inverted parabola that has a maximum at ( 0.5 , 1.0 ) (0.5,1.0) (0.5,1.0). If you draw the line y = x y=x y=x it intersects the parabola at ( 0 , 0 ) (0,0) (0,0) and ( 0.75 , 0.75 ) (0.75,0.75) (0.75,0.75). This line is useful for following the iterations or orbits of the function.
Start with a point on the x − x- x−axis, say x 0 = 0.2 x_0 = 0.2 x0=0.2 (point A). Find the value of F ( 0.2 ) = 4 × 0.2 × ( 1 − 0.2 ) = 0.64 F(0.2) = 4 \times 0.2 \times (1 - 0.2) = 0.64 F(0.2)=4×0.2×(1−0.2)=0.64. This point B is on the parabola above point A.
We want to iterate F ( x ) F(x) F(x) by using the y − y- y−coordinate of B as the next input. We could calculate F ( 0.64 ) = 4 × 0.64 × ( 1 − 0.64 ) = 0.9216 F(0.64) = 4 \times 0.64 \times (1 - 0.64) = 0.9216 F(0.64)=4×0.64×(1−0.64)=0.9216 and we'll want to write a Julia function to do that, but it's also useful to see how the orbit evolves.
Draw a horizontal line until you get to the line y = x y=x y=x at point C. Because C is on the 45 ° 45 \degree 45° line, then x C = y C x_C = y_C xC=yC so the coordinates are C = ( 0.64 , 0.64 ) C = (0.64,0.64) C=(0.64,0.64). The x − x- x−coordinate of C is the y − y- y−coordinate of B, so C becomes the next iterate. From C, draw a vertical line until you intersect the parabola at D which has coordinates ( 0.64 , 0.9216 ) (0.64,0.9216) (0.64,0.9216).
This process can be repeated as often as you like and will show the trajectory of the function F ( x ) F(x) F(x).
An Iterator for the Logistic Function¶
Like the iter_f function above, we can write a Julia function to iterate the logistic function. The inputs will be the starting point, x 0 x_0 x0, the constant c c c, and the number of iterations n n n with a default value of n = 100 n=100 n=100.
This new function, iter_logistic will return the orbit, y y y.
Let's try an example with x 0 = 0.2 x_0 = 0.2 x0=0.2
and c = 4 c = 4 c=4.
Now we can plot the trajectory:
Try adjusting the value for c c c. When c = 4 c=4 c=4 the plot seems pretty chaotic. What happens if you change x 0 x_0 x0 from 0.2 0.2 0.2 to 0.19 0.19 0.19? This should show that the trajectory is sensitive to the initial value of x 0 x_0 x0.
Next, try c = 0.5 c = 0.5 c=0.5 and set x 0 = 0.1 x_0 = 0.1 x0=0.1. Is the plot chaotic? Is it sensitive to initial conditions?
Autocorrelation and Orbits¶
While the plot of the logistic function trajectory for c = 4 c=4 c=4 appears chaotic, you might be able to convince yourself that there are repeating patterns. The repetitions are far from identical, but if you imagine making a copy of the plot and shifting it over a bit, it might line up.
This is what autocorrelation does. The amount of the shift is controlled by a lag τ \tau τ. For each y [ t ] y[t] y[t] autocorrelation compares it to y [ t − τ ] y[t - \tau] y[t−τ]. The output of the autocorrelation function is a number r k r_k rk between − 1 -1 −1 and 1 1 1, where
r k = ∑ t = τ + 1 N ( y [ t ] − μ ) ( y [ t − τ ] − μ ) ∑ t = 1 N ( y [ t ] − μ ) 2 r_k = \frac{\sum_{t = \tau + 1}^N (y[t] - \mu)(y[t - \tau] - \mu)}{\sum_{t=1}^N (y[t] - \mu)^2} rk=∑t=1N(y[t]−μ)2∑t=τ+1N(y[t]−μ)(y[t−τ]−μ)and μ \mu μ is the average of all the y y y's. If r k = 1 r_k = 1 rk=1 then y [ t ] = y [ t + τ ] y[t] = y[t + \tau] y[t]=y[t+τ] for every t t t, r k = 0 r_k = 0 rk=0 means there is no correlation, and r k = − 1 r_k = -1 rk=−1 says that y [ k ] = − y [ k + τ ] y[k] = -y[k + \tau] y[k]=−y[k+τ].
A orbit is the plot of y [ k ] y[k] y[k] on the x − x- x−axis and y [ k + τ ] y[k+\tau] y[k+τ] on the y − y- y−axis.
Let's start by plotting the autocorrelation of the iterated logistic function for different values of τ \tau τ.
Rather than scrolling back up to set x 0 x_0 x0 and c c c, let's just create two new variables and call them x P xP xP and c P cP cP.
Choose the autocorrelation step τ \tau τ:
The autocorrelation plot seems about as chaotic as the logistic function, but when τ 0 = 1 \tau_0 = 1 τ0=1, the phase portrait is very smooth. In fact, the outline of the blue dots ( y 1 y1 y1) looks just like the logistic function!
This should be expected since y n + 1 = f ( y n ) y_{n+1} = f(y_n) yn+1=f(yn) for every n n n.
Why choose τ 0 = 1 \tau_0 = 1 τ0=1? That choice of τ 0 \tau_0 τ0 plots the orbit of the function. The orange lines (y2) show which point follows the current location. On the left side, the path seems to be mostly upwards towards a nearby point. On the right, points are sent back to the left half except those near x = 1 x = 1 x=1.
Try setting τ 0 = 0 \tau_0 = 0 τ0=0 and you'll see that the orbit becomes the line y = x y=x y=x, because there's perfect correlation between y [ t ] y[t] y[t] and y [ t + τ 0 ] y[t + \tau_0] y[t+τ0]. The plot doesn't provide any insight. Try some other values of τ 0 \tau_0 τ0 to see what happens.
With Pluto, you can change the input values to a function to immediately see the effects.
Conclusion: Making Sense of the Unpredictable¶
Through our exploration with Pluto notebooks, we've seen how even simple mathematical rules can generate surprisingly complex behavior. We started with linear equations, where small changes in initial conditions lead to predictable differences in outcomes. Then, by making a slight modification to create the logistic function, we discovered how chaos emerges - where tiny differences in starting points can lead to wildly different results.
The key insights we've uncovered include:
- Linear systems behave predictably, with differences growing in a steady, controlled manner
- The logistic function, despite its simple form, can produce chaotic behavior when the parameter c is set appropriately
- Chaos doesn't mean random - the phase portraits revealed beautiful structure within the apparent disorder
- Interactive tools like Pluto notebooks allow us to experiment and visualize these complex dynamics directly
While chaos might seem to make prediction impossible, understanding chaos theory actually helps us better grasp the nature of complex systems - from weather patterns to population dynamics to heart rhythms. We now know that some systems have fundamental limits to predictability, not because of our ignorance or lack of precision, but because of their inherent sensitivity to initial conditions.
If you're interested in exploring further, try experimenting with:
- Different values of the parameter c c c in the logistic function
- Various initial conditions to see how trajectories diverge
- Other nonlinear functions to discover new types of chaotic behavior
- Applying these concepts to real-world data sets
Remember: chaos isn't about things being completely unpredictable - it's about understanding the delicate balance between order and randomness that governs many of the complex systems in our world.
Image credits¶
Hero image from Introduction to Online Nonstochastic Control, Elad Hazan, Karan Singh, Research Gate, Figure 2.7: Lorenz attractor Nov 2022.
Code and Software¶
EasyChaos.jl - A Pluto notebook to do some simple experiments with chaos theory.
- Julia - The Julia Project as a whole is about bringing usable, scalable technical computing to a greater audience: allowing scientists and researchers to use computation more rapidly and effectively; letting businesses do harder and more interesting analyses more easily and cheaply.
- Pluto.jl - Pluto is an environment to work with the Julia programming language. Easy to use like Python, fast like C.
Further Reading¶
- Bollt, E. M. (2022, March 2). How we can make sense of chaos. Quanta Magazine.
- National Institute of Standards and Technology (NIST). (n.d.). Autocorrelation plot.
- Thompson, J. M. T., & Stewart, H. B. (2002). General principles of chaotic dynamics. Journal of Mathematical Physics, 31(3), 332–343.
- Williams, G. P. (1997). Chaos theory simply explained. ResearchGate.
- Strogatz, S. H. (2018). Nonlinear dynamics and chaos: With applications to physics, biology, chemistry, and engineering (2nd ed.). Westview Press.
- Lorenz, E. N. (1963). Deterministic nonperiodic flow. Journal of the Atmospheric Sciences, 20(2), 130–141.
- Ott, E. (2002). Chaos in dynamical systems (2nd ed.). Cambridge University Press.
- Peitgen, H.-O., Jürgens, H., & Saupe, D. (2004). Chaos and fractals: New frontiers of science.
📬 Subscribe and stay in the loop. · Email · GitHub · Forum · Facebook
© 2020 – 2025 Jan De Wilde & John Peach