Генерация свежих имен для безымянных лямбда-терминов

Есть ли какой-то общий метод или библиотека для генерации свежих имен при преобразовании безымянных лямбда-терминов в именованные?

Это то, что я придумал (минимальный пример основан на представлении школы Haskell):

{-# LANGUAGE FlexibleInstances #-}

module Main where

import Control.Monad.Reader
import Data.List ((\\))

-- named terms -----------------------------------------------------------------

infixl 9 :@
infixr 0 :!

data Lambda a =
      Var a
    | (Lambda a) :@ (Lambda a)
    | a :! (Lambda a)
    deriving Show

-- nameless terms --------------------------------------------------------------

newtype Scope a = Scope (Exp a)
    deriving (Eq, Show, Read)

infixl 9 :$

data Exp a =
      F a
    | B !Int
    | (Exp a) :$ (Exp a)
    | L (Scope a)
    deriving (Eq, Show)

abstract :: Eq a => a -> Exp a -> Scope a
abstract name expr = Scope $ absName 0 expr where
    absName distance (F name') | name == name' = B distance
                               | otherwise        = F name'
    absName distance (B index) = B index
    absName distance (fun :$ arg) = absName distance fun :$ absName distance arg
    absName distance (L (Scope body)) = L $ Scope $ absName (succ distance) body

instantiate :: Exp a -> Scope a -> Exp a 
instantiate ersatz (Scope body) = replace 0 body where
    replace distance (B index) | index == distance = ersatz
                               | otherwise         = B index
    replace distance (F name) = F name
    replace distance (fun :$ arg) = replace distance fun :$ replace distance arg
    replace distance (L (Scope body)) = L $ Scope $ replace (succ distance) body

-- unnaming and naming ---------------------------------------------------------

unname :: Eq a => Lambda a -> Exp a
unname (Var a) = F a
unname (fun :@ arg) = unname fun :$ unname arg
unname (n :! body) = L (abstract n (unname body))

name :: (Eq a, Generator a) => Exp a -> Lambda a
name exp = runReader (nm exp) []

nm :: (Eq a, Generator a) => Exp a -> Reader [a] (Lambda a)
nm (F a) = return $ Var a
nm (fun :$ arg) = (:@) <$> nm fun <*> nm arg
nm (L scope) = do
    used <- ask
    let fresh = generate used
    undefined
    local (fresh:) ((fresh :!) <$> nm (instantiate (F fresh) scope))

type Gen a = [a]

class Eq a => Generator a where
    generator :: Gen a
    generate :: [a] -> a
    generate used = head (generator \\ used)

instance Generator [Char] where
    generator = [ [i] | i <- ['a'..'z']] ++ [i: show j | j <- [1..], i <- ['a'..'z']]

-- test data -------------------------------------------------------------------

s = "f" :! "g" :! "x" :! (Var "f" :@ Var "x") :@ (Var "g" :@ Var "x")
k = "x" :! "y" :! Var "x"
i = "x" :! Var "x"

Хотя это работает, я думаю, что в целом name, nm, generator конструкция не очень элегантная, и я уверен, что она не очень хорошо масштабируется. Есть ли общая техника или библиотека, которая используется в подобных случаях?

Спасибо заранее!

0 ответов

Другие вопросы по тегам