Learn more about Israeli genocide in Gaza, funded by the USA, Germany, the UK and others.

Proving 1+1=2 and other advanced theorems

What are numbers? What does it mean to add them? And why does a+b equal b+a? In this interactive session we’ll play the Natural Number Game to explore the foundations of math in Lean. Prerequisites: ability to count.

Motivation 1: can you convince a simpleton that a+b = b+a?

Imagine this dialogue between you and Simplicius the simpleton:

You: 2+3=3+2, wouldn’t you agree?

Simplicius: Hmm, I don’t see it.

You: Okay, 2+3=5 by calculation. And 3+2=5 by calculation. So both sides equal 5.

Simplicius: Okay, I see that. And it seems to work for some other examples.

You: Great. So you can see that, for any a and b, a+b=b+a?

Simplicius: Hmm, no, I don’t see that. How do you know?

You: ... C’mon, it’s obvious!

What can you say to Simplicius? Can you construct an argument that would convince him? That is, a proof?

But can’t we just say, “Simplicius is a rare idiot; let’s move on”? Let’s see a second example, which might be more familiar.

Motivation 2: convincing the computer your code is correct

You’ve written some code that gets the first number in an array:

function head(arr: number[]): number {
  if (arr.length === 0) {
    return -1;
  } else {
    return arr[0];
  }
}

Your type : number claims that this function always returns a number. I’m convinced. But annoyingly, the computer complains:

Type 'number | undefined' is not assignable to type 'number'

Stupid computer! Apparently it can’t see that, if the length of the array is not zero, there must be a value at the first index. Why not, and how can we prove it to the computer?

Our computers are like Simplicius. We need a formal argument to convince it that arr[0] really is a number.

Lean

Enter Lean. In its own words:

Lean is a functional programming language [... and] a theorem prover.

How could this help us solve our problems above?

The natural number game

In this session we’ll play The Natural Number Game. Click it and you should see:

Natural number game

We’ll play the first 8 levels. At the end, we will have proved that a+b=b+a.

1.1: rfl, reflection, “anything is equal to itself”

The most fundamental tactic, rfl! Sounds useless, but it’s almost a definition of equality!

example ( x y z : ℕ ) :
  x * y + z = x * y + z
:= by
  rfl

1.2: rw, rewriting, “If a=b, then you can replace a with b

A problem from primary school:

\[ \begin{aligned} a &= 3 \\ x &= 2 \times a \end{aligned} \]

What is the value of x? You might solve it by substituting:

\[ \begin{aligned} x &= 2 \times a \\ &= 2 \times 3 \\ &= 6 \end{aligned} \]

This is exactly what the rw tactic does.

example ( x y : ℕ ) ( h : y = x + 7 ) :
  2 * y = 2 * ( x + 7 )
:= by
  rw [h]
  rfl

1.3: 0 and succ, defining the natural numbers

Above we vaguely referred to “numbers”, but for the rest of this session, we’re only going to be using the “natural numbers”, 0, 1, 2, 3, ..., denoted ℕ. Other kinds of “number” (e.g. negatives, fractions, reals, complex, ...) don’t exist in this session!

We’ve been writing numbers in decimal, e.g. 12. But decimal is kinda complex, so here we’ll work with a simpler notation: 125 = 1*(10^2) + 2*(10^1) + 5*(10^0)

succ stands for “successor”. You can interpret it as “The number after n”. So instead of writing “3”, we instead write succ(succ(succ(0))).

Exercise: how would you write “7” in this notation?

Exercise: what number does succ(succ(succ(succ(0)))) represent?

Coders — you can think of the natural numbers as a linked list structure:

type Nat = 0 | { succ: Nat };
const three = { succ: { succ: { succ: 0 } } };

Onto the exercise:

example ( a b : ℕ ) ( h : ( succ a ) = b ) :
  succ ( succ a ) = succ b
:= by
  rw [h]
  rfl

1.4: introducing addition

To convince anyone that “addition is commutative”, we first need to say what we mean by “addition”!

Here is how the Natural Number Game defines addition:

add_zero (a : ℕ)   :  a + 0 = a
add_succ (a b : ℕ) :  a + (succ b) = succ (a + b)

Or in English: “To calculate a+b, repeatedly move the succs from b onto the outside, until there are none left on b. When there are none left, you have the total.”

Seeing this in action:

              succ(0) + succ(succ(0))
=      succ(  succ(0) +      succ(0)  )      // add_succ
= succ(succ(  succ(0) +           0   ))     // add_succ
= succ(succ(  succ(0)                 ))     // add_zero

Exercise: what numbers are we adding? And what is the result?

For the coders, here’s a rough equivalent in TypeScript:

function add(a: Nat, b: Nat): Nat {
  if (b === 0) {
    return a;
  } else {
    return { succ: add(a, b.succ) };
  }
}
example :
  a + succ 0 = succ a
:= by
  rw [add_succ]
  rw [add_zero]
  rfl

Interlude: induction

Imagine a chess board with a single bishop on one of the black squares. I then repeatedly move the bishop around (following normal rules: diagonal moves). What color square will the bishop be on after move 846,245?

Here’s an argument that it will be on a black square:

This is an argument by induction:

2.1: zero_add and induction n

So, we have a + 0 = a. We have it because we assumed it as an axiom that we called add_zero.

But what about 0 + a = a? Looks very similar — but is it the same?

No! It’s a separate claim, and we’ll call it zero_add. Our task here is to prove it.

You could try using rfl or rw [h]. But you won’t get very far. The final tactic you need is induction, written induction n in Lean.

Here’s the induction argument in English:

Here it is in Lean:

theorem MyNat.zero_add ( n : ℕ ) :
  0 + n = n
:= by
  induction n
  rw [add_zero]
  rfl
  rw [add_succ]
  rw [n_ih]
  rfl

2.2: add_assoc

From here, you don’t need to learn any new concepts! The last three levels can all be solved with combinations of rfl, rw, and induction. Good luck!

theorem MyNat.add_assoc ( a b c : ℕ ) :
  ( a + b ) + c = a + ( b + c )
:= by
  induction c
  rw [add_zero]
  rw [add_zero]
  rfl
  rw [add_succ]
  rw [add_succ]
  rw [add_succ]
  rw [n_ih]
  rfl

2.3: succ_add

theorem MyNat.succ_add ( a b : ℕ ) :
  succ a + b = succ ( a + b )
:= by
  induction b
  rw [add_zero]
  rw [add_zero]
  rfl
  rw [add_succ]
  rw [n_ih]
  rw [add_succ]
  rfl

2.4: add_comm

theorem MyNat.add_comm ( a b : ℕ ) :
  a + b = b + a
:= by
  induction b
  rw [add_zero]
  rw [zero_add]
  rfl
  rw [add_succ]
  rw [n_ih]
  rw [succ_add]
  rfl
Tagged #math, #programming.

Similar posts

More by Jim

Want to build a fantastic product using LLMs? I work at Granola where we're building the future IDE for knowledge work. Come and work with us! Read more or get in touch!

This page copyright James Fisher 2023. Content is not associated with my employer. Found an error? Edit this page.