summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authormarkus <unknown>2003-11-18 08:46:47 +0000
committermarkus <unknown>2003-11-18 08:46:47 +0000
commit6d34e007d936a12746f02d3ebc047184a9e7f992 (patch)
tree242dba362ff8327e2b0e57fbc2d42f9befa77007 /src
parent70c9f7b365b07044c07837a04223a11dfa3b7140 (diff)
polymorph trie added
Diffstat (limited to 'src')
-rw-r--r--src/GF/Data/Trie2.hs55
1 files changed, 55 insertions, 0 deletions
diff --git a/src/GF/Data/Trie2.hs b/src/GF/Data/Trie2.hs
new file mode 100644
index 000000000..e15c53e8c
--- /dev/null
+++ b/src/GF/Data/Trie2.hs
@@ -0,0 +1,55 @@
+{-
+ **************************************************************
+ * Author : Markus Forsberg *
+ * markus@cs.chalmers.se *
+ **************************************************************
+-}
+module Trie2 (
+ tcompile,
+ collapse,
+ Trie,
+ trieLookup
+ ) where
+
+import Map
+
+newtype TrieT a b = TrieT ([(a,TrieT a b)],[b])
+
+newtype Trie a b = Trie (Map a (Trie a b), [b])
+
+emptyTrie = TrieT ([],[])
+
+optimize :: Ord a => TrieT a b -> Trie a b
+optimize (TrieT (xs,res)) = Trie ([(c,optimize t) | (c,t) <- xs] |->+ empty,
+ res)
+
+collapse :: Ord a => Trie a b -> [([a],[b])]
+collapse trie = collapse' trie []
+ where collapse' (Trie (map,(x:xs))) s = if (isEmpty map) then [(reverse s,(x:xs))]
+ else (reverse s,(x:xs)):
+ concat [ collapse' trie (c:s) | (c,trie) <- flatten map]
+ collapse' (Trie (map,[])) s
+ = concat [ collapse' trie (c:s) | (c,trie) <- flatten map]
+
+tcompile :: Ord a => [([a],[b])] -> Trie a b
+tcompile xs = optimize $ build xs emptyTrie
+
+build :: Ord a => [([a],[b])] -> TrieT a b -> TrieT a b
+build [] trie = trie
+build (x:xs) trie = build xs (insert x trie)
+ where
+ insert ([],ys) (TrieT (xs,res)) = TrieT (xs,ys ++ res)
+ insert ((s:ss),ys) (TrieT (xs,res))
+ = case (span (\(s',_) -> s' /= s) xs) of
+ (xs,[]) -> TrieT (((s,(insert (ss,ys) emptyTrie)):xs),res)
+ (xs,(y,trie):zs) -> TrieT (xs ++ ((y,insert (ss,ys) trie):zs),res)
+
+trieLookup :: Ord a => Trie a b -> [a] -> ([a],[b])
+trieLookup trie s = apply trie s s
+
+apply :: Ord a => Trie a b -> [a] -> [a] -> ([a],[b])
+apply (Trie (_,res)) [] inp = (inp,res)
+apply (Trie (map,_)) (s:ss) inp
+ = case map ! s of
+ Just trie -> apply trie ss inp
+ Nothing -> (inp,[])