123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395 |
- #
- #
- # Nim's Runtime Library
- # (c) Copyright 2012 Andreas Rumpf
- #
- # See the file "copying.txt", included in this
- # distribution, for details about the copyright.
- #
- ## This module implements a `crit bit tree`:idx: which is an efficient
- ## container for a sorted set of strings, or for a sorted mapping of strings. Based on the excellent paper
- ## by Adam Langley.
- ## (A crit bit tree is a form of `radix tree`:idx: or `patricia trie`:idx:.)
- include "system/inclrtl"
- type
- NodeObj[T] = object {.acyclic.}
- byte: int ## byte index of the difference
- otherbits: char
- case isLeaf: bool
- of false: child: array[0..1, ref NodeObj[T]]
- of true:
- key: string
- when T isnot void:
- val: T
- Node[T] = ref NodeObj[T]
- CritBitTree*[T] = object ## The crit bit tree can either be used
- ## as a mapping from strings to
- ## some type ``T`` or as a set of
- ## strings if ``T`` is void.
- root: Node[T]
- count: int
- {.deprecated: [TCritBitTree: CritBitTree].}
- proc len*[T](c: CritBitTree[T]): int =
- ## returns the number of elements in `c` in O(1).
- result = c.count
- proc rawGet[T](c: CritBitTree[T], key: string): Node[T] =
- var it = c.root
- while it != nil:
- if not it.isLeaf:
- let ch = if it.byte < key.len: key[it.byte] else: '\0'
- let dir = (1 + (ch.ord or it.otherBits.ord)) shr 8
- it = it.child[dir]
- else:
- return if it.key == key: it else: nil
- proc contains*[T](c: CritBitTree[T], key: string): bool {.inline.} =
- ## returns true iff `c` contains the given `key`.
- result = rawGet(c, key) != nil
- proc hasKey*[T](c: CritBitTree[T], key: string): bool {.inline.} =
- ## alias for `contains`.
- result = rawGet(c, key) != nil
- proc rawInsert[T](c: var CritBitTree[T], key: string): Node[T] =
- if c.root == nil:
- new c.root
- c.root.isleaf = true
- c.root.key = key
- result = c.root
- else:
- var it = c.root
- while not it.isLeaf:
- let ch = if it.byte < key.len: key[it.byte] else: '\0'
- let dir = (1 + (ch.ord or it.otherBits.ord)) shr 8
- it = it.child[dir]
- var newOtherBits = 0
- var newByte = 0
- block blockX:
- while newbyte < key.len:
- let ch = if newbyte < it.key.len: it.key[newbyte] else: '\0'
- if ch != key[newbyte]:
- newotherbits = ch.ord xor key[newbyte].ord
- break blockX
- inc newbyte
- if newbyte < it.key.len:
- newotherbits = it.key[newbyte].ord
- else:
- return it
- while (newOtherBits and (newOtherBits-1)) != 0:
- newOtherBits = newOtherBits and (newOtherBits-1)
- newOtherBits = newOtherBits xor 255
- let ch = if newByte < it.key.len: it.key[newByte] else: '\0'
- let dir = (1 + (ord(ch) or newOtherBits)) shr 8
- var inner: Node[T]
- new inner
- new result
- result.isLeaf = true
- result.key = key
- inner.otherBits = chr(newOtherBits)
- inner.byte = newByte
- inner.child[1 - dir] = result
- var wherep = addr(c.root)
- while true:
- var p = wherep[]
- if p.isLeaf: break
- if p.byte > newByte: break
- if p.byte == newByte and p.otherBits.ord > newOtherBits: break
- let ch = if p.byte < key.len: key[p.byte] else: '\0'
- let dir = (1 + (ch.ord or p.otherBits.ord)) shr 8
- wherep = addr(p.child[dir])
- inner.child[dir] = wherep[]
- wherep[] = inner
- inc c.count
- proc exclImpl[T](c: var CritBitTree[T], key: string) : int =
- var p = c.root
- var wherep = addr(c.root)
- var whereq: ptr Node[T] = nil
- if p == nil: return c.count
- var dir = 0
- var q: Node[T]
- while not p.isLeaf:
- whereq = wherep
- q = p
- let ch = if p.byte < key.len: key[p.byte] else: '\0'
- dir = (1 + (ch.ord or p.otherBits.ord)) shr 8
- wherep = addr(p.child[dir])
- p = wherep[]
- if p.key == key:
- # else: not in tree at all
- if whereq == nil:
- c.root = nil
- else:
- whereq[] = q.child[1 - dir]
- dec c.count
- return c.count
- proc excl*[T](c: var CritBitTree[T], key: string) =
- ## removes `key` (and its associated value) from the set `c`.
- ## If the `key` does not exist, nothing happens.
- discard exclImpl(c, key)
- proc missingOrExcl*[T](c: var CritBitTree[T], key: string): bool =
- ## Returns true iff `c` does not contain the given `key`. If the key
- ## does exist, c.excl(key) is performed.
- let oldCount = c.count
- var n = exclImpl(c, key)
- result = c.count == oldCount
- proc containsOrIncl*[T](c: var CritBitTree[T], key: string, val: T): bool =
- ## returns true iff `c` contains the given `key`. If the key does not exist
- ## ``c[key] = val`` is performed.
- let oldCount = c.count
- var n = rawInsert(c, key)
- result = c.count == oldCount
- when T isnot void:
- if not result: n.val = val
- proc containsOrIncl*(c: var CritBitTree[void], key: string): bool =
- ## returns true iff `c` contains the given `key`. If the key does not exist
- ## it is inserted into `c`.
- let oldCount = c.count
- var n = rawInsert(c, key)
- result = c.count == oldCount
- proc inc*(c: var CritBitTree[int]; key: string, val: int = 1) =
- ## increments `c[key]` by `val`.
- var n = rawInsert(c, key)
- inc n.val, val
- proc incl*(c: var CritBitTree[void], key: string) =
- ## includes `key` in `c`.
- discard rawInsert(c, key)
- proc incl*[T](c: var CritBitTree[T], key: string, val: T) =
- ## inserts `key` with value `val` into `c`.
- var n = rawInsert(c, key)
- n.val = val
- proc `[]=`*[T](c: var CritBitTree[T], key: string, val: T) =
- ## puts a (key, value)-pair into `t`.
- var n = rawInsert(c, key)
- n.val = val
- template get[T](c: CritBitTree[T], key: string): T =
- let n = rawGet(c, key)
- if n == nil:
- when compiles($key):
- raise newException(KeyError, "key not found: " & $key)
- else:
- raise newException(KeyError, "key not found")
- n.val
- proc `[]`*[T](c: CritBitTree[T], key: string): T {.inline, deprecatedGet.} =
- ## retrieves the value at ``c[key]``. If `key` is not in `t`, the
- ## ``KeyError`` exception is raised. One can check with ``hasKey`` whether
- ## the key exists.
- get(c, key)
- proc `[]`*[T](c: var CritBitTree[T], key: string): var T {.inline,
- deprecatedGet.} =
- ## retrieves the value at ``c[key]``. The value can be modified.
- ## If `key` is not in `t`, the ``KeyError`` exception is raised.
- get(c, key)
- proc mget*[T](c: var CritBitTree[T], key: string): var T {.inline, deprecated.} =
- ## retrieves the value at ``c[key]``. The value can be modified.
- ## If `key` is not in `t`, the ``KeyError`` exception is raised.
- ## Use ```[]``` instead.
- get(c, key)
- iterator leaves[T](n: Node[T]): Node[T] =
- if n != nil:
- # XXX actually we could compute the necessary stack size in advance:
- # it's roughly log2(c.count).
- var stack = @[n]
- while stack.len > 0:
- var it = stack.pop
- while not it.isLeaf:
- stack.add(it.child[1])
- it = it.child[0]
- assert(it != nil)
- yield it
- iterator keys*[T](c: CritBitTree[T]): string =
- ## yields all keys in lexicographical order.
- for x in leaves(c.root): yield x.key
- iterator values*[T](c: CritBitTree[T]): T =
- ## yields all values of `c` in the lexicographical order of the
- ## corresponding keys.
- for x in leaves(c.root): yield x.val
- iterator mvalues*[T](c: var CritBitTree[T]): var T =
- ## yields all values of `c` in the lexicographical order of the
- ## corresponding keys. The values can be modified.
- for x in leaves(c.root): yield x.val
- iterator items*[T](c: CritBitTree[T]): string =
- ## yields all keys in lexicographical order.
- for x in leaves(c.root): yield x.key
- iterator pairs*[T](c: CritBitTree[T]): tuple[key: string, val: T] =
- ## yields all (key, value)-pairs of `c`.
- for x in leaves(c.root): yield (x.key, x.val)
- iterator mpairs*[T](c: var CritBitTree[T]): tuple[key: string, val: var T] =
- ## yields all (key, value)-pairs of `c`. The yielded values can be modified.
- for x in leaves(c.root): yield (x.key, x.val)
- proc allprefixedAux[T](c: CritBitTree[T], key: string; longestMatch: bool): Node[T] =
- var p = c.root
- var top = p
- if p != nil:
- while not p.isLeaf:
- var q = p
- let ch = if p.byte < key.len: key[p.byte] else: '\0'
- let dir = (1 + (ch.ord or p.otherBits.ord)) shr 8
- p = p.child[dir]
- if q.byte < key.len: top = p
- if not longestMatch:
- for i in 0 ..< key.len:
- if p.key[i] != key[i]: return
- result = top
- iterator itemsWithPrefix*[T](c: CritBitTree[T], prefix: string;
- longestMatch=false): string =
- ## yields all keys starting with `prefix`. If `longestMatch` is true,
- ## the longest match is returned, it doesn't have to be a complete match then.
- let top = allprefixedAux(c, prefix, longestMatch)
- for x in leaves(top): yield x.key
- iterator keysWithPrefix*[T](c: CritBitTree[T], prefix: string;
- longestMatch=false): string =
- ## yields all keys starting with `prefix`.
- let top = allprefixedAux(c, prefix, longestMatch)
- for x in leaves(top): yield x.key
- iterator valuesWithPrefix*[T](c: CritBitTree[T], prefix: string;
- longestMatch=false): T =
- ## yields all values of `c` starting with `prefix` of the
- ## corresponding keys.
- let top = allprefixedAux(c, prefix, longestMatch)
- for x in leaves(top): yield x.val
- iterator mvaluesWithPrefix*[T](c: var CritBitTree[T], prefix: string;
- longestMatch=false): var T =
- ## yields all values of `c` starting with `prefix` of the
- ## corresponding keys. The values can be modified.
- let top = allprefixedAux(c, prefix, longestMatch)
- for x in leaves(top): yield x.val
- iterator pairsWithPrefix*[T](c: CritBitTree[T],
- prefix: string;
- longestMatch=false): tuple[key: string, val: T] =
- ## yields all (key, value)-pairs of `c` starting with `prefix`.
- let top = allprefixedAux(c, prefix, longestMatch)
- for x in leaves(top): yield (x.key, x.val)
- iterator mpairsWithPrefix*[T](c: var CritBitTree[T],
- prefix: string;
- longestMatch=false): tuple[key: string, val: var T] =
- ## yields all (key, value)-pairs of `c` starting with `prefix`.
- ## The yielded values can be modified.
- let top = allprefixedAux(c, prefix, longestMatch)
- for x in leaves(top): yield (x.key, x.val)
- proc `$`*[T](c: CritBitTree[T]): string =
- ## turns `c` into a string representation. Example outputs:
- ## ``{keyA: value, keyB: value}``, ``{:}``
- ## If `T` is void the outputs look like:
- ## ``{keyA, keyB}``, ``{}``.
- if c.len == 0:
- when T is void:
- result = "{}"
- else:
- result = "{:}"
- else:
- # an educated guess is better than nothing:
- when T is void:
- const avgItemLen = 8
- else:
- const avgItemLen = 16
- result = newStringOfCap(c.count * avgItemLen)
- result.add("{")
- when T is void:
- for key in keys(c):
- if result.len > 1: result.add(", ")
- result.addQuoted(key)
- else:
- for key, val in pairs(c):
- if result.len > 1: result.add(", ")
- result.addQuoted(key)
- result.add(": ")
- result.addQuoted(val)
- result.add("}")
- when isMainModule:
- import sequtils
- var r: CritBitTree[void]
- r.incl "abc"
- r.incl "xyz"
- r.incl "def"
- r.incl "definition"
- r.incl "prefix"
- r.incl "foo"
- doAssert r.contains"def"
- r.excl "def"
- assert r.missingOrExcl("foo") == false
- assert "foo" notin toSeq(r.items)
- assert r.missingOrExcl("foo") == true
- assert toSeq(r.items) == @["abc", "definition", "prefix", "xyz"]
- assert toSeq(r.itemsWithPrefix("de")) == @["definition"]
- var c = CritBitTree[int]()
- c.inc("a")
- assert c["a"] == 1
- c.inc("a", 4)
- assert c["a"] == 5
- c.inc("a", -5)
- assert c["a"] == 0
- c.inc("b", 2)
- assert c["b"] == 2
- c.inc("c", 3)
- assert c["c"] == 3
- c.inc("a", 1)
- assert c["a"] == 1
- var cf = CritBitTree[float]()
- cf.incl("a", 1.0)
- assert cf["a"] == 1.0
- cf.incl("b", 2.0)
- assert cf["b"] == 2.0
- cf.incl("c", 3.0)
- assert cf["c"] == 3.0
- assert cf.len == 3
- cf.excl("c")
- assert cf.len == 2
|