Skip to content

Hindley-Milner型推論を実装してみる

License

Notifications You must be signed in to change notification settings

nojima/typed-lambda

Folders and files

NameName
Last commit message
Last commit date

Latest commit

7171b10 · Jan 10, 2025

History

89 Commits
Apr 9, 2020
Aug 22, 2020
Aug 22, 2020
Apr 22, 2020
Mar 28, 2020
Apr 10, 2020
Mar 28, 2020
Aug 22, 2020
Aug 22, 2020
Mar 28, 2020
Mar 28, 2020
Aug 22, 2020
Jan 10, 2025
Jan 10, 2025
Aug 22, 2020

Repository files navigation

型付きラムダ計算

標準入力に渡された項を解釈し、型検査して実行します。 Hindley-Milner 型推論を実装しているので、プログラマが型を明示的に指定しなくても型検査ができようになっています。

stack run < samples/sample1.txt

文法

この言語に存在する型は以下の3つのみです。

Int       -- 整数
Bool      -- Boolean
T1 -> T2  -- T1からT2への関数

リテラル

自然数リテラルとBoolリテラルがあります。

0
100
true
false

ラムダ式

ラムダ式は以下のような構文です。 引数は必ず一つだけです。

lambda x. x

引数を複数個取るような関数が欲しいときはラムダをネストさせてください。

lambda f.
    lambda n.
        f n

関数適用

項を並べれば関数適用になります。関数適用は左結合です。

(lambda x. x * x) 100

if式

以下のように if ... then ... else ... という構文です。

if true then 10 else 20

let式

let name = expr in ... という構文で式を変数に束縛できます。

let x = 10 in x * x

ラムダを束縛すると名前付き関数の宣言として使えます。

let f = lambda x.
    x * x
in
f 100

let多相も実装されています。

let identity = lambda x. x in
let n = identity 100 in
let b = identity true in
if b then n else n+1

let多相はその名の通りletで束縛したときに起こります。 以下のように identity を lambda の引数として渡すと identity は単相な型として型付けされるので、TypeError になります。

(lambda identity.
    let n = identity 100 in
    let b = identity true in
    if b then n else n+1
) (lambda x. x)

def式

def name arg = expr in ... という構文で関数を定義できます。 この構文で定義された関数は再帰呼び出しが可能です。

def fact n =
  if n == 0 then
    1
  else
    n * fact (n-1)
in
fact 5

List

[x1, x2, ..., xn] という構文でリストを定義できます。 また、以下の関数がリストに対して使えます。

関数名 説明
head List[a] -> a リストの先頭を返す。リストが空なら実行時エラー。
tail List[a] -> List[a] リストの先頭以外の要素を返す。リストが空なら実行時エラー。
null List[a] -> Bool リストが空なら true を返し、そうでないなら false を返す。
cons a -> List[a] -> List[a] 第二引数の先頭に第一引数を付け加えたリストを返す。

例:

def map f =
  lambda list.
    if null list then
      []
    else
      let x  = head list in
      let xs = tail list in
      cons (f x) (map f xs)
in

def square x =
  x * x
in

map square [1, 2, 3, 4, 5]

Tuple

(x1, x2, ..., xn) という構文でタプルを生成できます。 また、以下の関数がタプルに対して使えます。

関数 説明
fst Tuple[a, b] -> a ペアの1番目の要素を返す。
snd Tuple[a, b] -> b ペアの2番目の要素を返す。

パターンマッチ

match ... with ... という構文でパターンマッチができます。

例:

match (true, 10, 20) with
  | (false, x, y) -> x - y
  | (true,  x, y) -> x + y

※ 今のところリストのパターンマッチには対応していません。

二項演算子

シンボル 意味 結合
+ 足し算 Int -> Int -> Int
- 引き算 Int -> Int -> Int
* 掛け算 Int -> Int -> Int
/ 割り算(小数点以下切り捨て) Int -> Int -> Int
&& 論理積 Bool -> Bool -> Bool
|| 論理和 Bool -> Bool -> Bool
== 等しい ∀a. a -> a -> Bool

※ 論理和を全角文字で表記していますが、正しくは半角です。 表の中に | を書くと GitHub の Markdown のパースが狂うので全角で書いています。

コメント

-- による行コメントと {- ... -} という範囲コメントが使えます。

About

Hindley-Milner型推論を実装してみる

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published