跳表(skiplist)在redis/levelDB中属于核心数据结构,我简单粗暴的用Golang实现了下。
就我的简单理解来说,就一个普通的链表,在insert时,通过Random_level(),把一层变成很多层,
越上数据越小,跨度越大。 查找时从上往下找,如果在一层没找到,在下一层继续时,以此节点作为起始,
继续查找,是一种用空间换时间的方式。
测试代码:
package main //SkipList //author:Xiong Chuan Liang //date:2014-1-28 import ( "github.com/xcltapestry/xclpkg/algorithm" ) func main() { slt := algorithm.NewSkipList() for i := 100; i > 0; i-- { slt.Insert(i) } slt.PrintSkipList() slt.Search(15) slt.Search(93) slt.Remove(93) slt.PrintSkipList() }运行效果 :
SkipList------------------------------------------- level: 7 -------------------------------------------------------- level: 6 -------------------------------------------------------- level: 5 -------------------------------------------------------- level: 4 -------------------------------------------------------- level: 3 93 -------------------------------------------------------- level: 2 36 93 -------------------------------------------------------- level: 1 6 10 22 23 25 36 42 47 48 54 55 61 72 75 78 82 89 93 -------------------------------------------------------- level: 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 -------------------------------------------------------- Current MaxLevel: 4 Search() Level= 3 93 Search() Level= 2 36 Search() Level= 1 6 10 22 Search() Level= 0 11 12 13 14 15 Found level= 0 key= 15 Search() Level= 3 93 Found level= 3 key= 93 Remove() level= 3 key= 93 Remove() level= 2 key= 93 Remove() level= 1 key= 93 Remove() level= 0 key= 93 SkipList------------------------------------------- level: 7 -------------------------------------------------------- level: 6 -------------------------------------------------------- level: 5 -------------------------------------------------------- level: 4 -------------------------------------------------------- level: 3 -------------------------------------------------------- level: 2 36 -------------------------------------------------------- level: 1 6 10 22 23 25 36 42 47 48 54 55 61 72 75 78 82 89 -------------------------------------------------------- level: 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 94 95 96 97 98 99 100 -------------------------------------------------------- Current MaxLevel: 3
从上面可以很清楚的看到构成及查找过程.
跳表的结构体:
type Node struct { Forward []Node Value interface{} } type SkipList struct { Header *Node Level int }与普通List不同,因为跳表在从上层移到下层查找时,要跳过上层已遍历过的Node,所以,要求在下一层时,立马转到对应位置,因此在Node中定
义了Forward数组. 最初实现时,我使用分层次的普通List来做,构建,删除Node啥的都很方便,但发现查找怎么实现都不太对,才发现思路错了,
只好重构改成现在的方式.效果还比较理想.
实现代码放在Github上. https://github.com/xcltapestry/xclpkg/tree/master/algorithm
欢迎指正.
另外,图灵社区关于跳表的两篇文章相当不错: http://www.ituring.com.cn/article/59053
MAIL: [email protected]
BLOG: http://blog.csdn.net./xcl168