第9章 查找

9.1 静态查找表

  • 查找操作的性能分析
    平均查找长度(Average Search Length)ASL =\mathop \sum_{i=1}^{n} P_i C_i。其中P_i为查找表中第i个记录出现的概率,C_i为在查找到第i个记录之前访问过的记录数。一般我们可以根据给定的P_i设计出合理的查找顺序以使ASL最小。

9.1.1 顺序表的查找

顺序查找(Sequential Search):省略,没什么好说的。

9.1.2 有序表的查找

  1. 折半查找/二分查找(Binary Search)
    • 折半查找(二分查找)的性能分析
      描述查找过程的二叉树称为判定树。判定树一般不为满二叉树,但是根据二分法的原理,其叶子节点所在层次之差不超过1(不大于1)。所以具有n个结点的判定树的深度和具有n个结点的完全二叉树的深度相同。则具有n个结点的判定树的深度为floor( \log_2n)+1
      为方便讨论,假设有序查找表的长度n = 2^h-1,则描述折半查找(二分查找)的判定树是深度为h的满二叉树。
      当查找表中每个记录的查找概率相等且均为\dfrac{1}{n}时(P_i = \frac{1}{n}),查找成功的平均查找长度为ASL_{success} = \sum_{i=1}^{n} P_i C_i = \frac{1}{n} \sum_{j=1}^{h} j \times 2^{j-1} = \frac{n+1}{n} \log_2(n+1) - 1
      由此可见,折半查找的效率比顺序查找的效率高,但是折半查找只适用于有序表,并且仅适用于顺序存储结构,对于链式存储结构的线性链表无法有效进行折半查找。
  2. Fibonacci查找
  3. 黄金分割查找(参见《运筹学教程》)
    • 黄金分割搜索是针对单峰函数的,可以用于关键字的值先升后降(或先降后升)的有序查找表的查找。
  4. 插值查找:对于关键字的值分布较为均匀的查找表(记录的结合),我们也可以采用插值查找。即根据当前查找范围(scope)中关键字的值的上下限预先估计出在当前查找范围中开始查找的下标。

9.1.3 静态树表的查找

当有序表中各记录的查找概率不等时,应如何构造判定树使得查找性能最优?
若只考虑查找成功的情况,使得PH = \sum_{i=1}^{n} \omega_i h_i的值取得最小值的判定树为查找性能最佳的判定树,称为静态最优查找树(Static Optimal Search Tree)。其中n为有序查找表的长度,同时也是判定树上结点的个数;结点的权\omega_i与结点的查找概率p_i成正比或直接取p_i(考虑到计算机的整数运算性能较高,也可以适当选择缩放因子使得所有的\omega_i都为整数);h_i为第i个结点在二叉树上的层数(应该从0开始还是从1开始?这个好像无所谓,因为影响只是使得PH值相差了\sum_{i=1}^{n} \omega_i,这个对于静态查找表是常量)。

次优查找树(Nearly Optimal Search Tree)的构造方法

//todo

9.1.4 索引顺序表的查找

分块查找又称为索引顺序查找,是顺序查找的一种改进方法。除查找表本身以外,尚需要建立一个“索引表”。将查找表分成最大记录数相同的若干子表(最大记录数为预先设定),或将查找表均匀地等分为给定数量的子表。对每个子表(块)建立一个索引项,其中包括两项内容:关键字项(其值为该子表内关键字的最大值)指针项(指向该子表中第一个记录在表中的位置)
其实,分块查找如何进行是不言自明的。索引表中的指针项,对于顺序存储结构的查找表可以存储其这一子表第一个元素在主查找表中的下标(偏移量),对于链式存储结构则可以存储指向这一子表的第一个结点的指针。有一个数据结构就利用了多级索引表,这就是跳表,redis的实现中用到了跳表。

9.2 动态查找表

动态查找表结构本身是在查找过程中动态生成的。

9.2.1 二叉排序树(Binary Search Tree)和平衡二叉树

  1. 二叉排序树及其查找过程
    二叉排序树(Binary Search Tree)定义省略,二叉排序树又称为二叉查找树。
  2. 二叉排序树的插入和删除
    插入过程相当显然,但是删除过程就不易理解。以待删除结点的直接前驱(直接后继)代替这一待删除结点,而后从待删除结点的左子树(右子树)中删去这一直接前驱(直接后继)
namespace binary_sort_tree
    {
        typedef int DataType;//在C++中这个可以用重载代替

        typedef struct Node_s
        {
            DataType data;
            struct Node_s *left, *right;
        } Node_t;
        
        //中序遍历输出以root为根的数的值
        inline void display(Node_t *root,std::ostream &outputStream)
        {
            if (root == NULL) return;
            if (root->left != NULL) display(root->left,outputStream);
            outputStream << root->data<<"\t";
            if (root->right != NULL) display(root->right, outputStream);
        }

        inline void insert(const DataType &data, Node_t *&root)
        {
            if (root == NULL)
            {
                //生成新的结点
                root = reinterpret_cast<Node_t *>(malloc(sizeof(Node_t)));
                root->data = data;
                root->left = root->right = NULL;
                return;
            }
            else if (data == root->data)
            {//如果等于,那么就用新的替换旧的
                root->data = data;
                return;
            }
            else
            {
                //否则就在其左子树或者右子树上
                if (data < root->data)
                {//要插入到左结点的
                    if (root->left != NULL) insert(data, root->left);
                    else
                    {
                        //生成新的结点
                        root->left = reinterpret_cast<Node_t *>(malloc(sizeof(Node_t)));
                        root->left->data = data;
                        root->left->left = root->left->right = NULL;
                    }
                }
                if (data > root->data)
                {//要插入到右结点的
                    if (root->right != NULL) insert(data, root->right);
                    else
                    {
                        //生成新的结点
                        root->right = reinterpret_cast<Node_t *>(malloc(sizeof(Node_t)));
                        root->right->data = data;
                        root->right->left = root->right->right = NULL;
                    }
                }
            }
        }

        //在以root为根的排序二叉树中
        //尝试找到该结点并删除之
        inline void remove(const DataType &data,Node_t *&root)
        {
            
            if (root == NULL) return;
            if (data < root->data)//说明可能在左子树中
            {
                if (root->left != NULL) remove(data, root->left);
                else return;
            }
            if (data > root->data)//说明可能在右子树中
            {
                if (root->right != NULL) remove(data, root->right);
                else return;
            }
            if (data == root->data)
            {
                //当前root指针所指的结点即为
                if (root->left == NULL || root->right == NULL)
                {
                    if (root->left == NULL && root->right == NULL)
                    {
                        free(root);
                        root = NULL;
                    }
                    else
                    {
                        if (root->left != NULL && root->right == NULL)
                        {
                            Node_t *newRoot = root->left;
                            free(root);
                            root = newRoot;
                        }
                        if (root->left == NULL && root->right != NULL)
                        {
                            Node_t *newRoot = root->right;
                            free(root);
                            root = newRoot;
                        }
                    }
                }
                else
                {
                    //两边子树都非空
                    //这就是最复杂的情况了
                    //方法1:找到前驱结点
                    if (root->left->right == NULL)
                    {
                        //此时前驱结点即为root->left
                        root->left->right = root->right;
                        free(root);
                        root = root->left;
                    }
                    else
                    {
                        Node_t *p, *q;
                        for (p = root->left,q = root->left->right;
                            q->right != NULL; p = q, q = q->right);
                        //此时前驱结点即为q,前驱结点的双亲结点即为p
                        p = q->left;//此时就从树中摘除了前驱结点q
                        //但是q的内存区域尚未释放,事实上,将要作为新的结点
                        q->left = root->left;
                        q->right = root->right;
                        root = q;
                        
                    }
                    //方法2:找到后继结点
                }
            }
        }

        inline void test()
        {
            //插入结点
            Node_t *tree = NULL;
            insert(45, tree);
            insert(12, tree);
            insert(53, tree);
            insert(3, tree);
            insert(37, tree);
            insert(100, tree);
            insert(24, tree);
            insert(61, tree);
            insert(90, tree);
            insert(78, tree);

            display(tree, std::cout);

            std::cout << std::endl;

            remove(90, tree);
            display(tree, std::cout);

            std::cout << std::flush;
        }
    }
  1. 二叉排序树的查找分析
  2. 平衡二叉树
    平衡二叉树(Balanced Binary Tree或Height-Balanced Tree)或是一棵空树,或其左右子树的深度差不超过1,且其左右子树均为平衡二叉树。平衡二叉树又称为AVL树。
    结点的平衡因子(Balance Factor,BF):该结点左子树与右子树的深度之差。则平衡二叉树上所有结点的平衡因子只能为-1或0或1。
  • 左旋与右旋
    首先定义针对一棵树的左旋与右旋操作,设指针a指向一棵树的根节点。
    1. 左旋:在左旋之后,新的根节点为*(a->right),设b = a->right为指向旋转之后新的根节点的指针(旋转操作所需要进行的指针的修改此处不表)。*b的右子树为旋转操作前*a的右子树的右子树;*b的左子树为以旋转前的*a为根,以旋转前*a的左子树作为左子树,以旋转前*a的右子树的左子树作为右子树的一颗二叉树。如下表所示。
      旋转前 旋转后
      根节点 a a->right
      左子树 a->left ROOT:a;
      LEFT:a->left;RIGHT:a->right->left
      右子树 a->right a->right->right
    2. 右旋:在右旋之后,新的根节点为*(a->left),设b = a->left为指向旋转之后新的根节点的指针(旋转操作所需要进行的指针的修改此处不表)。*b的左子树为旋转操作前*a的左子树的左子树;*b的右子树为以旋转前的*a为根,以旋转前*a的左子树的右子树作为左子树,以旋转前*a的右子树作为右子树的一颗二叉树。如下表所示。
      旋转前 旋转后
      根节点 a a->left
      左子树 a->left a->left->left
      右子树 a->right ROOT:a;
      LEFT:a->left->right;RIGHT:a->right
    其实左旋和右旋在二叉树的抽象数据结构的层面上是相当形象和易于理解的(配图省略)。并且左旋和右旋互为对称(互逆)。由于旋转之前的二叉树为排序二叉树BST,不难证明该旋转操作后得到的以*b为根的树仍符合排序二叉树的定义,依然是一颗排序二叉树。
  • 如何使得二叉排序树为平衡二叉树呢?需要对不平衡的二叉树进行“旋转”。在初始时,二叉树为空树,也是平衡的排序二叉树。在插入结点使得排序二叉树失去平衡时,设指针a指向离新插入结点最近且不平衡(平衡因子绝对值超过1)的祖先节点。可以根据新插入结点位于*a的左子树或右子树进行如下分类讨论。
    插入结点位置 *a的BF变化
    *a的左子树 {1} \rightarrow{2}
    *a的右子树 {-1} \rightarrow{-2}
    1. 当新插入的结点位于*a的左子树且导致以*a作为根的二叉树失去平衡时,*a的BF值由1变为2。此时应当尝试右旋以使二叉树再平衡。
      在进行右旋之后得到的以*b为根的排序二叉树是否平衡呢?我们需要讨论以*b为根的树的左右子树深度之差。
      由于此时*a的BF值从1变为2,我们可以设插入这一结点之前*a的左子树的深度为h*a的右子树的深度为h-1。插入这一结点之后*a的左子树的深度增加到了h+1。且由于a指向离新插入结点最近且不平衡(平衡因子绝对值超过1)的祖先节点,则*a的左右子树都是平衡的。
      根据右旋操作的定义,旋转之后得到的排序二叉树(Binary Sort Tree)的右子树的深度不仅取决于旋转之前*(a->right)的深度(h-1),也取决于旋转之前*a的左子树的右子树*(a->left->right);左子树即为旋转之前左子树的左子树*(a->left->left)。因此,为讨论右旋之后得到的以*b为根的排序二叉树的左右子树深度之差(以考察该树是否平衡),我们需要对新插入的这一结点位于*a的左子树或是右子树中进行分类讨论。
      1. 若新的结点插入到了*a的左子树的左子树中,并使得*a左子树的深度由h增加到了h+1。则说明插入这一结点使得*a的左子树的左子树的深度从h-1增加到了h,且*a的左子树的右子树的深度为h-1(根据指针a的定义,*a的左子树*(a->left)在插入新结点前后都是平衡的)。
        深度
        a->left->left h
        a->left->right h-1
        a->right h-1
        在右旋之后,*b的左子树(旋转之前*a的左子树的左子树)的深度为h*b的右子树的左子树(旋转之前*a的左子树的右子树)的深度为h-1,而*b的右子树的右子树(旋转之前*a的右子树)的深度为h-1,故*b的右子树的深度为h
        旋转后 深度
        *b的左子树 a->left->left h
        *b的右子树 ROOT:a;
        LEFT:a->left->right;
        RIGHT:a->right
        \max \{ {h-1},{h-1} \} +1 = h
        此时以*b为根的排序二叉树的BF值为0,是平衡排序二叉树。
      2. 若新的结点插入到了*a的左子树的右子树中,并使得*a左子树的深度由h增加到了h+1。则说明插入这一结点使得*a的左子树的右子树的深度从h-1增加到了h,且*a的左子树的左子树的深度为h-1(根据指针a的定义,*a的左子树*(a->left)在插入新结点前后都是平衡的)。
        深度
        a->left->left h-1
        a->left->right h
        a->right h-1
        在右旋之后,*b的左子树(旋转之前*a的左子树的左子树)的深度为h-1*b的右子树的左子树(旋转之前*a的左子树的右子树)的深度为h,而*b的右子树的右子树(旋转之前*a的右子树)的深度为h-1,故*b的右子树的深度为h+1
        旋转后 深度
        *b的左子树 a->left->left h-1
        *b的右子树 ROOT:a;
        LEFT:a->left->right;
        RIGHT:a->right
        \max \{ h,h-1 \} +1 = h+1
        此时以*b为根的排序二叉树的BF值为-2。该二叉树不平衡,我们需要进行进一步的处理。
        根据上文分析得到的插入新的结点之后,右旋之前各个子树的深度。
        将新结点插入到*(a->left->left) 将新结点插入到*(a->left->right)
        a->left->left h h-1
        a->left->right h-1 h
        a->right h-1 h-1
        此时容易想到,如果在将新的结点插入到*a的左子树的右子树之后,若能进行某种操作使得以*(a->left->left)为根的子树的深度从h-1变为h,使得以*(a->left->right)为根的子树的深度从h变为h-1,在此之后就与上述将新的结点插入到*a的左子树的左子树的情况相同。(对于将排序二叉树再平衡的操作,只需维持排序二叉树的性质,新插入结点的位置无需关心)
        容易证明,在对以*a为根的树进行右旋之前,*a的左子树进行左旋即可满足上述要求。(具体怎么样证明我实在不想写了,向上面那样讨论其与左旋有关的各个子树即可)
    2. 当新插入的结点位于*a的右子树且导致以*a作为根的二叉树失去平衡时,*a的BF值由-1变为-2。此时应当尝试左旋以使二叉树再平衡。

9.2.2 B-树和B+树

  1. B-树及其查找
    B-树是一种平衡的多路查找树,其在文件系统中很有用。
    • 一棵m阶的B-树,或为空树,或为满足下列条件的m叉树:
      1. 树中每个结点至多有m棵子树
      2. 根结点至少有两颗子树(可能为空树)
      3. 除根结点之外的所有非空结点至少有floor(\frac{m}{2})(其中floor函数为向下取整)棵子树。
      4. 所有非空结点包含如下信息数据
        (n,A_0,K_1,A_1,K_2,A_2,\cdots,K_n,A_n)
        其中\{ K_i \} (i=1,2,\cdots,n)序列为有序关键字序列,A_i为指向子树根节点的指针,且满足\forall K_i(i=1,2,\cdots,n),均有K_i大于A_{i-1}所指向子树中的所有关键字,且K_i小于A_i所指向子树中的所有关键字。n为本结点中关键字数量(n+1为本结点中子树数量)。
        空树被视为查找失败的结点(外部结点),事实上这些结点不存在,指向这些空树的指针为空。
  2. B-树查找分析
  3. B-树的插入和删除结点
    • 在最底层某个结点中添加一个关键字,以及与相邻关键字间隔的空子树。添加完成后若该结点的关键字个数达到m,即该结点的字数数量达到了m+1。既达到了该结点容量的上限(无法插入更多关键字),也不再符合B-树“每个结点至多有m棵子树”的要求,故将要产生结点的“分裂”。
    • 在删除关键字时,若被删除关键字所在的结点的关键字数量少于floor(m/2),则应当与兄弟节点进行合并。
  4. B+树
    我看教材上没怎么介绍,而且已经不符合树的定义,在此就省略吧。

9.2.3 键树(数字查找树,Digital Search Trees)

键树是一颗多叉树,其中每一个结点中存储的是组成关键字的符号。由此可见,键树往往不是通用的,而是domain-specified的。并且在具体应用时,可能需要设定特定的叶子结点作为关键字(值)的结束,这样的叶子节点中可能需要一定的标志位,亦或是约定的(不可能作为关键字的一部分的)符号;这样的末尾叶子结点中可能还会带有对应于该关键字(的值)的记录,亦或是指向该记录的指针。

  • 通常,键树可有两种存储结构
    设组成关键字的符号数量为d(包括一个代表关键字结束的符号),即所有关键字的可行值均可由d-1个互不相同的符号组成。
    1. 以树的孩子兄弟链表表示
    2. 以树的多重链表表示(此时键树也可称为Trie树)
      每个结点中应包含有d个指针域…………Trie树是很直观也很容易设计的。

9.2 Hash表

9.3.1 什么是Hash表

Hash函数f(K)的定义从略。
对不同的关键字可能得到同一Hash地址,这种现象称为冲突(collision)。在一般情况下冲突只能尽可能少,而不能完全避免,因为从实用的角度考虑,Hash函数是一个从关键字集合到地址集合的压缩映射。因此建立哈希表(散列)时不仅要设定一个尽可能避免冲突且产生的地址尽可能短的“好”的Hash函数,也要设定一种处理冲突的方法。
均匀的(Uniform)的Hash函数:通俗地说,对于给定的关键字集合,经Hash函数产生的各个地址值的概率相等。

9.3.2 Hash函数的构造方法

  1. 直接定址法:取关键字或关键字的某个一次函数值为Hash地址
    显然,由于Hash地址值是关键字的线性映射,所以不会发生冲突。当然这样做意义不大,实际应用中不常用。
  2. 数字分析法(这种方法需要针对关键字集合具体分析,得到Hash函数)
  3. 平方取中法:取关键字平方后的中间几位(由地址表长确定)为Hash地址
  4. 折叠法(folding)
    将关键字分割成位数相同的几部分(最后一部分的位数可以不同),然后取这几部分的叠加和(无论进位考虑与否均可)作为Hash地址。
  5. 除留余数法
    取关键字被某个不大于Hash表表长的数p整除后所得的余数为Hash地址,即key % p。值得注意的是对p的选择很重要。

9.3.3 处理冲突的方法

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351

推荐阅读更多精彩内容