> A purely academic programming language based on lambda calculus and De
> Bruijn indices written in Haskell.
[Jump to examples](#Examples) or use the navigation tree to jump to
other sections.
Docs, articles, examples and more:
[website](https://bruijn.marvinborner.de).
## Features
- **De Bruijn indices[\[0\]](#References)** eliminate the complexity of
α-equivalence and α-conversion
- Unique **bracket-style representation** for lambda abstractions
enables improved human-readability and faster syntactic perception
- **Call-by-need** reduction with great time/memory complexity by using
the RKNL[\[4\]](#References) abstract machine (similar to
[calm](https://github.com/marvinborner/calm/))
- **Syntactic sugar** for unary/binary/ternary numerals and
binary-encoded strings and chars
- **No primitive functions** - every function is implemented in Bruijn
itself
- Highly space-efficient compilation to **binary lambda calculus
(BLC)[\[2\]](#References)[\[3\]](#References)** additionally to normal
interpretation and REPL
- Strongly **opinionated parser** with strict syntax rules
- **Recursion** can be implemented using combinators such as Y, Z or ω
- Substantial **standard library** featuring many useful functions (see
`std/`)
## Basics
### De Bruijn indices
De Bruijn indices[\[0\]](#References) replace the concept of variables
in lambda calculus. The index basically represents the abstraction layer
you want to reference beginning at 0 with the innermost layer.
For example, λx.x becomes λ0 because x referenced the first abstraction
layer. Furthermore, λx.λy.xy becomes λλ10 and so forth.
You can read more about De Bruijn indices on
[Wikipedia](https://en.wikipedia.org/wiki/De_Bruijn_index).
### Syntax
In general the syntax of bruijn is pretty similar to the previously
presented normal lambda calculus syntax with De Bruijn indices. The main
difference of the syntax of expressions is the usage of square brackets
instead of λs (e.g. `[[1 0]]` instead of λλ10).
You can use any function that you’ve previously defined. You can also
overwrite previously defined functions. The environment gets interpreted
from bottom to top (starting at `main`).
The following are the main syntax specifications in the (minorly
extended) [Backus-Naur
form](https://en.wikipedia.org/wiki/Backus%E2%80%93Naur_form)
(prefix/infix/mixfix operators are omitted for simplicity).
::= [a-ω,A-Ω,_][a-ω,A-Ω,0-9,?,!,',-]*
::= [A-Ω][a-ω,A-Ω]+
::= "[" "]"
::= ("+" | "-")[0-9]+[u|b|t]
::= [0-9]
::= | | | "(" ")" | [namespace.] ::= ::= | ::= ":test " "(" ") (" ")"
::= ":import " [namespace]
::= ":input " ::= "# " *
Differences in syntax between REPL and file:
**For files**:
The execution of a file begins at the `main` function. Its existence is
mandatory. Note the missing equal sign in definitions.
::= ::= | | | | "\n"
**For REPL**:
::= = ::= ":watch "