Wednesday, October 30, 2024

14 - Disquality Again

We've just seen how a lemma can be applied to a proof goal. 

Here we'll see how lemmas can be applied to hypotheses too.


Task

Given a natural number n<5, show that

n5

This is the very same task as the previous post.


Maths

We're going to use the same lemma we used in the previous chapter. For natural numbers a and b

a<bab

Previously we considered the proof goal n5, and used this lemma to say that proving n<5 was sufficient.

This time we'll consider the hypothesis n<5, and say that n5 follows directly as a result of that lemma.

The following diagram makes clear this difference. We can see the lemma builds half a bridge from our hypothesis. Do compare it to the diagram from the previous post.

Let's write a proof that applies the lemma to the hypothesis.

(1)n<5hypothesis(2)n5proof objective (3)a<babexisting lemma(4)n5lemma (3) applied to (1)n<5n5by lemma (3)

Again, this may look a little over-done, but the small steps will help us write a Lean proof. 

  • We start with the hypothesis n<5, and our proof objective n5
  • We know about a lemma (3) applicable to natural numbers, that if a<b then ab
  • The lemma's antecedent a<b matches our hypothesis (1), which immediately gives us n5.

Let’s explain some terminology. In an implication PQ, P is called the antecedent, and Q the consequent.


Code

The following Lean program proves that a natural number n5, given n<5.


-- 14 - Lemma: Not Equal from Less Than

import Mathlib.Tactic

example {n : ℕ} (h : n < 5) : n ≠ 5 := by
  apply ne_of_lt at h
  exact h

The Lean proof is almost exactly the same as the previous post's proof.

The only difference is the lemma ne_of_lt is pointed at the hypothesis h by adding at h.

This small change is significant. It changes the hypothesis h from h : n < 5 to h : n ≠ 5, as we'll see in the Infoview.

This new hypothesis exactly matches the proof goal, so exact h works just as before to complete the proof.


Infoview

Placing the cursor before apply ne_of_lt at h shows the original hypothesis.


n : ℕ
h : n < 5
⊢ n ≠ 5

Moving the cursor to the beginning of the next line after apply ne_of_lt at h shows the hypothesis has been replaced.


n : ℕ
h : n ≠ 5
⊢ n ≠ 5



Easy Exercise

Write a Lean program to prove n5, given n>5, where n is a natural number.

Use the same the lemma for “not equal from greater than” you found for the last post's exercise, and apply it to the hypothesis.


Tuesday, October 22, 2024

Appendix A - Taxonomy

The following diagram is a simplified hierarchy of the objects in Lean (click to enlarge).

There are two top-level objects called universes, Type and Prop.

  • Within the Type universe, there are sets, such as the familiar N, Z and R. Each of these sets has elements, such as 13.
  • Within the Prop universe, there are propositions, such as Odd 13. Some of these propositions may have proofs

Under the top level universes, the second level objects are called types. The third level objects are called terms.


Unsurprising Example

When we write 13 : ℕ, we mean “13, the element of the set of natural numbers ℕ”. 

We also say 13 is a term of type .


Surprising Example

When we write h : Odd 13, we mean “h, the proof of the proposition Odd 13”. 

The proposition Odd 13 is a type,  just as is a type.

Proofs like h or by use 6; norm_num could be terms of the type Odd 13, just as 13 is a term of type .

Rest assured, most people find this odd to begin with.


Tuesday, October 15, 2024

13 - Disquality

In addition to definitions, like Odd and Even, Mathlib also contains lemmas and theorems we can use.

We'll see how to use a lemma to support a simple disequality proof.


Task

Given a natural number n<5, show that

n5

The expression n5 is a disequality, whereas the hypothesis n<5 is an inequality.


Maths

It is common knowledge that, given two natural numbers a and b, if a<b is true, then we can say ab. That is,

a<bab

That common knowledge might seem too trivial to fuss over, but we'll think of it as a small lemma.

Looking at that lemma, we can see its conclusion ab matches our own proof objective n5. So, if we can show n<5, then we can conclude n5, using that lemma.

The following diagram illustrates how this lemma builds half a bridge to our proof objective.

Let's write a proof that uses this lemma.

(1)n<5hypothesisn5proof objective (2)a<babexisting lemma(3)n<5sufficient goal, by lemma (2)(4)n<5using (1)n<5n5by lemma (2)

This may look a little over-cooked, but the small steps will help us write a Lean proof. 

  • We start with the hypothesis n<5, and our proof objective n5
  • We know about a lemma (2) applicable to natural numbers, that if a<b then ab. So if we can prove n<5, then we can conclude n5
  • The lemma's conclusion matches our proof objective, so if we can prove n<5, then we can conclude n5.
  • This changes our proof goal from n5 to n<5
  • To prove n<5 is easy because we're given it as a hypothesis (1). 

So n<5, and by lemma (2) we finally conclude n5.


Code

The following Lean program proves that a natural number n5, given n<5.


-- 13 - Lemma: Not Equal from Less Than

import Mathlib.Tactic

example {n : ℕ} (h: n < 5): n ≠ 5 := by
  apply ne_of_lt
  exact h

The proof header declares n as a natural number, establishes the hypothesis h: n < 5, and specifies the proof objective n ≠ 5.

The apply instruction applies a lemma or theorem to the current goal, usually resulting in a change in goal.

Here, it applies a lemma named ne_of_lt, which means “not equal from less than”. It allows us to prove the current “not equal” goal by instead proving a “less than” goal.

The Infoview will show that apply ne_of_lt does indeed change the current proof goal from n ≠ 5 to n < 5.

The goal is now n < 5. We could use apply h to resolve it, but since the goal matches exactly the hypothesis h, we can instead use the instruction exact h

Notice that exact is applying a hypothesis here, not a lemma from Mathlib. The difference is not significant because both hypotheses and lemmas state facts.

It may be helpful to correlate this new, not yet familiar, code back to the maths proof. Here apply ne_of_lt corresponds to line (4) of the maths proof, and exact h corresponds to line (5).


Apply & Exact

We can use apply wherever we use exact. The benefit of exact is that it is stricter than apply

The hypothesis or lemma must exactly match the current goal, and if a misunderstanding has led to that not being true, it will be exposed immediately.


Infoview

Placing the cursor before apply ne_of_lt shows the original proof goal.


n : ℕ
h : n < 5
⊢ n ≠ 5

Moving the cursor to the beginning of the next line after apply ne_of_lt shows the goal has indeed changed.


n : ℕ
h : n < 5
⊢ n < 5


Lemmas & Theorems

The distinction between what is called a lemma or a theorem in Mathlib is not precise. Ultimately it doesn't matter as both are used in the same way.

Searching for suitable lemmas and theorems in Mathlib is currently not ideal. Many do conform to a naming convention, which helps. The exercise at the end provides an opportunity to practice finding a lemma using the naming convention.



Easy Exercise

Write a Lean program to prove n5, given n>5, where n is a natural number.

The proof will be almost exactly the same as this chapter's example, except the lemma will be “not equal from greater than”. 

Work out the required lemma's Mathlib name from the naming convention, or search the online Lean documentation to find it.


Saturday, October 12, 2024

12 - Odd & Even

In Part III we'll practise using lemmas and definitions in our proofs. 

There is a huge body of commonly agreed knowledge that mathematicians refer to in their own proofs. A large, and ever-growing, number of these lemmas, theorems, and definitions are encoded in the Mathlib library, ready for us to use in our own Lean proofs. 

We'll start by using the definition of an odd number.


Task

Show the integer 13 is odd.


Maths

To show 13 is odd, we need to show it meets the definition of odd.

An odd integer is of the form 2k+1, where k is an integer. 

In more mathematical words, if there exists an integer k such that n=2k+1, then n is odd.

The task has become an existence proof. If we can find an integer k such that 13=2k+1, then we have shown 13 is odd.

Let's write out a step by step proof.

13 is oddproof objective (1)kZ[n=2k+1]n is odddefinition of oddkZ[13=2k+1]sufficient goal, using (1)(2)use k=6chosen example13=2(6)+1using (2)13=2(6)+113 is oddby definition (1)

This may look a little laborious, but the detail will help us develop a Lean proof. . 

We start with the proof objective, to show 13 is odd.

We then state the definition, that n is odd if it can be written in the form 2k+1, where k is an integer. 

So, to show 13 is odd, it is sufficient to show it can be written in the form 2k+1. This gives us a new goal, to show there exists an integer k such that 13=2k+1.

We choose k=6, and confirm that 13=2(6)+1

We have shown that 13 can indeed be written in the form 2k+1. And so, by the definition of odd, we have finally shown that 13 is odd.


Code

The following Lean program proves the integer 13 is odd.


-- 12 - Definition: Odd Number

import Mathlib.Tactic

example : Odd (13: ℤ)  := by
  dsimp [Odd]
  use 6
  norm_num

The proof objective states that 13 is Odd

If we had simply written Odd 13 as the proof objective, 13 would be interpreted, by default, as a natural number. We write (13: ℤ) to specify 13 as an integer.

The idea of Odd is defined in Mathlib. The next line dsimp [Odd] expands that definition in the Infoview so we can see what it actually is. We'll see below the goal changes from being displayed as Odd 13, to ∃ k, 13 = 2 * k + 1, which we recognise as the definition of odd applied to 13.

The instruction dsimp has no effect on the proof itself. It only changes how Odd is displayed in the Infoview. 

After this point, the proof proceeds as a simple existence proof. 

The instruction use 6 tells Lean we want to try 6 for k. This changes the goal to 13 = 2 * 6 + 1. We resolve this goal by arithmetic, using the norm_num tactic. For such a simple and clear goal, there is no need for a multi-line calc section.


What is Odd?

The proof objective was written as Odd (13: ℤ). The intention is to say 13 is an odd number.

How does this work?

The idea of Odd is defined in Mathlib as a function. A definition is distinct from a lemma, and Mathlib contains many of both. A lemma needs to be justified by proof, but a definition does not.

That function Odd takes one parameter, and outputs a proposition involving that parameter, which may or may not be true.

When applied to 13, the output is a proposition  ∃k such that 13 = 2*k + 1. This proposition is true because we can prove it.

If we applied Odd to 14, the output is a proposition ∃k such that 14 = 2*k + 1. This proposition is not true.

It is interesting to see the actual definition of Odd inside Mathlib:


def Odd (a : α) : Prop := ∃ k, a = 2 * k + 1

We can see how a is mapped to a proposition about a.


Infoview

The Infoview is particularly useful when working with definitions and existence proofs.

Placing the cursor before dsimp [Odd] shows the original proof goal.


⊢ Odd 13

Moving the cursor to the end of the line after dsimp [Odd] shows the goal is now displayed using the definition of Odd.


⊢ ∃ k, 13 = 2 * k + 1

Placing the cursor after use 6 shows the goal is now specific to k = 6.


⊢ ∃ k, 13 = 2 * 6 + 1



Easy Exercise

Write a Lean program to prove the integer 14 is even.

Your proof should use Mathlib's definition Even for even numbers. Use dsimp to see how the definition is applied to 14.