2014-09-19 17:10:27 -07:00
|
|
|
* Lean declarations
|
|
|
|
|
|
|
|
** Definitions
|
|
|
|
|
2014-09-20 07:44:48 -07:00
|
|
|
The command =definition= declares a new constant/function. The identity function is defined as
|
2014-09-19 17:10:27 -07:00
|
|
|
|
|
|
|
#+BEGIN_SRC lean
|
2015-11-20 17:03:17 -08:00
|
|
|
definition identity {A : Type} (a : A) : A := a
|
2014-09-19 17:10:27 -07:00
|
|
|
#+END_SRC
|
|
|
|
|
2014-09-20 07:44:48 -07:00
|
|
|
We say definitions are "transparent", because the type checker can
|
|
|
|
unfold them. The following declaration only type checks because =+= is
|
|
|
|
a transparent definition. Otherwise, the type checker would reject
|
|
|
|
the expression =v = w= since it would not be able to establish that
|
|
|
|
=2+3= and =5= are "identical". In type theory, we say they are
|
|
|
|
_definitionally equal_.
|
2014-09-19 17:10:27 -07:00
|
|
|
|
|
|
|
#+BEGIN_SRC lean
|
2015-08-13 16:45:34 -07:00
|
|
|
import data.tuple data.nat
|
2014-09-19 17:10:27 -07:00
|
|
|
open nat
|
2015-08-13 16:45:34 -07:00
|
|
|
check λ (v : tuple nat (2+3)) (w : tuple nat 5), v = w
|
2014-09-19 17:10:27 -07:00
|
|
|
#+END_SRC
|
|
|
|
|
2015-11-20 17:03:17 -08:00
|
|
|
Similarly, the following definition only type checks because =identity= is transparent, and the type checker can establish that
|
2014-10-02 10:22:19 -07:00
|
|
|
=nat= and =id nat= are definitionally equal, that is, they are the "same".
|
2014-09-19 17:10:27 -07:00
|
|
|
|
|
|
|
#+BEGIN_SRC lean
|
|
|
|
import data.nat
|
2015-11-20 17:03:17 -08:00
|
|
|
definition identity {A : Type} (a : A) : A := a
|
|
|
|
check λ (x : nat) (y : identity nat), x = y
|
2014-09-19 17:10:27 -07:00
|
|
|
#+END_SRC
|
|
|
|
|
|
|
|
** Theorems
|
|
|
|
|
|
|
|
In Lean, a theorem is just an =opaque= definition. We usually use
|
2014-09-20 07:44:48 -07:00
|
|
|
=theorem= for declaring propositions. The idea is that users don't
|
2014-09-19 17:10:27 -07:00
|
|
|
really care about the actual proof, only about its existence. As
|
|
|
|
described in previous sections, =Prop= (the type of all propositions)
|
|
|
|
is _proof irrelevant_. If we had defined =id= using a theorem the
|
|
|
|
previous example would produce a typing error because the type checker
|
2015-11-20 17:03:17 -08:00
|
|
|
would be unable to unfold =identity= and establish that =nat= and =identity nat=
|
2014-10-02 10:22:19 -07:00
|
|
|
are definitionally equal.
|
2014-09-19 17:10:27 -07:00
|
|
|
|
|
|
|
** Private definitions and theorems
|
|
|
|
|
2014-09-20 07:44:48 -07:00
|
|
|
Sometimes it is useful to hide auxiliary definitions and theorems from
|
2014-09-19 17:10:27 -07:00
|
|
|
the module interface. The keyword =private= allows us to declare local
|
|
|
|
hidden definitions and theorems. A private definition is always
|
|
|
|
opaque. The name of a =private= definition is only visible in the
|
|
|
|
module/file where it was declared.
|
|
|
|
|
|
|
|
#+BEGIN_SRC lean
|
|
|
|
import data.nat
|
|
|
|
open nat
|
|
|
|
private definition inc (x : nat) := x + 1
|
|
|
|
private theorem inc_eq_succ (x : nat) : succ x = inc x :=
|
|
|
|
rfl
|
|
|
|
|
|
|
|
-- The definition inc and theorem inc_eq_succ are not visible/accessible
|
|
|
|
-- in modules that import this one.
|
|
|
|
#+END_SRC
|
|
|
|
|
|
|
|
** Protected definitions and theorems
|
|
|
|
|
|
|
|
Declarations can be be organized into namespaces. In the previous
|
|
|
|
examples, we have been using the namespace =nat=. It contains
|
|
|
|
definitions such as =nat.succ= and =nat.add=. The command =open=
|
|
|
|
creates the aliases =succ= and =add= to these definitions. An alias
|
|
|
|
will not be created for a _protected definition_ unless the user
|
|
|
|
explicitly request it.
|
|
|
|
|
|
|
|
#+BEGIN_SRC lean
|
|
|
|
import data.nat
|
|
|
|
open nat
|
|
|
|
namespace foo
|
|
|
|
definition two : nat := 2
|
|
|
|
protected definition three : nat := 3
|
|
|
|
end foo
|
|
|
|
open foo
|
|
|
|
check two
|
|
|
|
|
|
|
|
-- The following command produces a 'unknown identifier' error
|
|
|
|
/-
|
|
|
|
check three
|
|
|
|
-/
|
|
|
|
|
|
|
|
-- We have to use its fully qualified name to access three
|
|
|
|
check foo.three
|
|
|
|
|
|
|
|
-- If the user explicitly request three, then an alias is created
|
|
|
|
open foo (three)
|
|
|
|
check three
|
|
|
|
#+END_SRC
|