zoukankan      html  css  js  c++  java
  • 7.2 有序表查找

    先对线性表进行有序排列,之后进行查找

    主要有折半查找,插值查找和斐波那契查找

    它们的时间复杂度均是O(logn),显然远远好于顺序查找的O(n)。但就平均性能来说,斐波那契查找要优于折半查找。可惜如果是最坏的情况,key=1,那么始终都处于左侧长半区在查找,则查找效率要低于折半查找。

    还有比较关键的一点,折半查找是进行加法与除法运算,插值查找进行复杂的四则运算,而斐波那契查找只是进行最简单的加法运算,这种细微差别会影响最终的查找效率。

    下面我们就用php来描述这三种算法

    <?php
    header("content-type:text/html;charset=utf-8");
    /**
     *有序表查找操作
     *
     *包括
     * 1.初始化 __contruct()
     * 2.二分查找 binary_search()
     * 3.插值查找 insert_search()
     * 4.斐波那契查找 fibinacci_search()
    
     */
    class Order_search{
        private $a;
        private $length;
    
        //初始化
        public function __construct($a = array())
        {
            $this->length = count($a);
            for ($i = 1;$i<=$this->length;$i++){
                $this->a[$i] = $a[$i-1];
            }
    
    
        }
    
        //二分查找
        public function binary_search($key){
    
            $low = 1;                         //定义最低下标为记录首位
            $high = $this->length;           //定义最高下标为记录末位
            while ($low<=$high){
                $mid = intval(($low+$high)/2);//折半
    
                if($key<$this->a[$mid]){     //若查找值比中值小
                    $high = $mid - 1;        //最高下标调整到中值下标小一位
                }
                elseif ($key>$this->a[$mid]){//若查找值比中值大
                    $low = $mid + 1;          //最高下标调整到中值下标大一位
                }
                else
                    return $mid;            //若相等则说明mid即为查找到的位置
            }
            return 0;
        }
    
        //插值查找
    
        public function insert_search($key){
    
            $low = 1;
            $high = $this->length;
            while ($low<=$high){
                $mid = intval($low + ($high-$low) * ($key - $this->a[$low]) / ($this->a[$high] - $this->a[$low]));
    
                if($key<$this->a[$mid]){
                    $high = $mid - 1;
                }
                elseif ($key>$this->a[$mid]){
                    $low = $mid + 1;
                }
                else
                    return $mid;
            }
            return 0;
        }
    
        //斐波那契查找
        //为了实现斐波那契查找算法,我们首先要准备一个斐波那契数列
    
        function Fbi($i){
            if($i < 2){
                return ($i == 0 ? 0 : 1);
            }
            return $this->Fbi($i - 1) + $this->Fbi($i - 2);
        }
    
        function fibinacci_search($key){
            $low = 1;                                               //定义最低下标为记录首位
            $high = $this->length;                                 //定义最高下标为记录末位
            $k = 0;
            while ($this->length>$this->Fbi($k)-1){               //计算n位于斐波那契数列的位置
                $k++;
            }
            for ($i = $this->length;$i<$this->Fbi($k)-1;$i++){    //将不满的数值补全
                $this->a[$i] = $this->a[$this->length];
            }
            while ($low<=$high){
                $mid = $low + $this->Fbi($k-1)-1;                  //计算当前分隔的下标
                if($key<$this->a[$mid]){                          //若查找记录小于当前分隔记录
                    $high = $mid - 1;                             //最高下标调整到分隔下标mid-1处
                    $k = $k -1;                                   //斐波那契下标减一位
                }
                elseif ($key > $this->a[$mid]){                 //若查找记录大于当前分隔记录
                    $low = $mid + 1;                             //最高下标调整到分隔下标mid+1处
                    $k = $k -2;                                  //斐波那契下标减2位
                }
                else{
                    if($mid <= $this->length)
                        return $mid;                           //若相等则说明mid即为查找到的位置
                    else
                        return $this->length;                 //若mid>数组长度说明是补全数值,返回长度
                }
            }
            return 0 ;
        }
    
    
    }
    ?>

    实现上述函数:

    <?php
    header("content-type:text/html;charset=utf-8");
    include "order_search.class.php";
    
    $a = array(0,1,16,24,35,47,59,62,73,88,99);
    echo "初始化数组:";
    echo "</br>";
    $order_search = new Order_search($a);
    print_r($order_search);
    echo "</br>";
    echo "</br>";
    
    echo "二分查找显示62在数组中的位置:";
    echo "</br>";
    $key = $order_search->binary_search(62);
    echo $key;
    echo "</br>";
    echo "</br>";
    
    echo "插值查找显示73在数组中的位置:";
    echo "</br>";
    $key = $order_search->insert_search(73);
    echo $key;
    echo "</br>";
    echo "</br>";
    
    echo "斐波那契查找显示35在数组中的位置:";
    echo "</br>";
    $key = $order_search->fibinacci_search(35);
    echo $key;
    ?>

    最后的实现结果:

    三种查找算法本质上是分割点的选择不同,各有优劣。

  • 相关阅读:
    storm学习之入门篇(一)
    大数据基础知识:分布式计算、服务器集群[zz]
    weblogic 12c 配置jvm的内存大小
    JamCam创业故事:辞掉工作,去开发一个应用
    数据挖掘-关联规则分析[ZZ]
    十道海量数据处理面试题与十个方法大总结(转载)
    设计模式简单说明
    jQuery 设置select,radio的值,无法自动触发绑定的change事件
    切换node版本,node-sass安装报错
    jQuery之编写插件
  • 原文地址:https://www.cnblogs.com/xlzfdddd/p/9893143.html
Copyright © 2011-2022 走看看