2016-03-24 15 views
2

Ich habe einen ausgeglichenen binären Baum, der Baum der Tiefe in seiner Art umfasst:Einzelpasslinse für die Wurzel eines ausgeglichenen Binärbaum

data Nat = Zero | Succ Nat 
data Tree (n :: Nat) a where 
    Leaf :: Tree Zero a 
    Branch :: a -> (Tree n a, Tree n a) -> Tree (Succ n) a 

ich einen Weg möchten beliebige Funktionen f :: Tree n a -> Tree n a auf der Unterstruktur laufen der Tiefe n an der Wurzel irgendeines Tree m a, mn.

ich in der Lage war, diese eine Möglichkeit, eine Typklasse zum Extrahieren und Ersetzen des Wurzelunterbaumes zu implementieren mit:

mapRoot :: X m n => (Tree n a -> Tree n a) -> Tree m a -> Tree m a 
mapRoot f t = putRoot (f (getRoot t)) t 

class X m n where 
    getRoot :: Tree m a -> Tree n a 
    putRoot :: Tree n a -> Tree m a -> Tree m a 

instance X m Zero where 
    getRoot t = Leaf 
    putRoot Leaf t = t 

instance X m n => X (Succ m) (Succ n) where 
    getRoot (Branch a (l,r)) = (Branch a (getRoot l, getRoot r)) 
    putRoot (Branch a (l,r)) (Branch _ (l',r')) = Branch a (putRoot l l', putRoot r r') 

Während dies funktioniert, bedarf es zwei Durchgänge durch den Wurzelunterbaum, und ich möchte um es in einem zu tun, wenn möglich.

Dies ist fast möglich durch faule Auswertung mit (den Bund fürs Leben):

mapRoot' :: Y m n => (Tree n a -> Tree n a) -> Tree m a -> Tree m a 
mapRoot' f t = t' where 
    (r, t') = swapRoot t r' 
    r' = f r 

class Y m n where 
    swapRoot :: (Tree m a, Tree n a) -> (Tree n a, Tree m a) 

instance Y m Zero where 
    swapRoot t leaf = (leaf, t) 

instance Y m n => Y (Succ m) (Succ n) where 
    swapRoot (Branch a (l,r)) (Branch a' (l',r')) = (Branch a (lx,rx), Branch a' (lx',rx')) where 
    (lx,lx') = swapRoot l l' 
    (rx,rx') = swapRoot r r' 

Aber wenn Sie tatsächlich versuchen, mapRoot' laufen Sie feststellen, dass es nicht zu stoppen; Das liegt daran, dass swapRoot in seinem zweiten Argument nicht faul ist (was nicht sein kann, weil Tree n a eine GADT ist).

jedoch gegeben getRoot und putRoot habe ich eine Linse für die Root-Teilstruktur bekommt, die ich führt zu vermuten, es gibt andere, darunter eine, die verwendet werden kann, mapRoot in einem einzigen Durchgang zu implementieren.

Was ist ein solches Objektiv?

+1

Darf ich fragen, warum Sie hier eine GADT anstelle eines verschachtelten Typs wählen? – dfeuer

+0

dfeuer: Ich denke, ich muss zur Kompilierzeit Behauptungen machen, die einfacher sind mit Naturals als mit verschachtelten Paaren ... aber ich werde darüber nachdenken. – rampion

+0

Eigentlich glaube ich, ich habe dich missverstanden. Wenn Sie "Teilbaum" sagen, sprechen Sie tatsächlich über den obersten Teil des Baumes (d. H. Alle Knoten bis zu einer bestimmten Tiefe)? – dfeuer

Antwort

3

Ihr Ansatz, den Knoten zu knüpfen, ist ein guter Klang - Sie brauchen nur alle Parameter an der richtigen Stelle, damit die Funktion ausreichend faul ist.

data (:<=) (n :: Nat) (m :: Nat) where 
    LTEQ_0 :: 'Zero :<= n 
    LTEQ_Succ :: !(n :<= m) -> 'Succ n :<= 'Succ m 

mapRoot :: n :<= m -> (Tree n a -> Tree n a) -> Tree m a -> Tree m a 
mapRoot p0 f0 t0 = restore (f0 root) where 
    (root, restore) = go p0 t0 

    go :: n :<= m -> Tree m a -> (Tree n a, Tree n a -> Tree m a) 
    go LTEQ_0 t = (Leaf, const t) 
    go (LTEQ_Succ p) (Branch a (l,r)) = 
    case (go p l, go p r) of 
     ((l', fl), (r', fr)) -> 
     (Branch a (l', r') 
     , \(Branch a1 (l1, r1)) -> Branch a1 (fl l1, fr r1) 
     ) 

anzumerken, dass go kehrt ein Paar - der Wurzelbaum, und eine Funktion des verarbeiteten Wurzel zu nehmen und um das Ergebnis zurück. Dies macht es explizit (zum Programm und zur Laufzeit!), Dass die resultierende Tree n a nicht von der Eingabe Tree n a abhängt.

Außerdem habe ich Ihre Klasse mit einem GADT nur der Kürze wegen ersetzt.

+1

Ich mag das besser als der Hack, den ich herausgefunden habe, um meine GADT mit 'swapRoot (Zweig a (l, r)) zu faulen anzupassen t = ... wo ~ (a ', (l', r ')) = (\ (Zweig ap) -> (a, p)) t 'da der Mangel an Abhängigkeit jetzt explizit ist. – rampion

+0

Das 'Case' sieht hier zu streng aus.Ich denke, es führt zu zwei Durchgängen (einer expliziten; einer implizit). – dfeuer

+0

@dfeuer Ich stimme zu, der Fall ist streng - aber wäre das in diesem Fall nicht erwünscht? Da "go" streng ist, muss es sowohl den Stammbaum als auch die Funktion berechnen, die das Endergebnis aus dem transformierten Baum auf einmal berechnet, was bedeutet, dass es nur einen Durchlauf machen muss. Vielleicht habe ich etwas falsch verstanden. – user2407038