>백엔드 개발 >PHP 튜토리얼 >PHP 순서 목록 검색----보간 검색

PHP 순서 목록 검색----보간 검색

黄舟
黄舟원래의
2016-12-28 09:46:181176검색

서문:

앞서 이진 검색을 소개했는데, 생각해보면 왜 반으로 해야 할까요? 4분의 1 이상으로 접는 것보다?

예를 들어 영어 사전에서 'apple'을 검색하면 사전을 펼칠 때 무의식적으로 앞 페이지로 넘어가나요, 아니면 뒷 페이지로 넘어가나요? "동물원"을 다시 확인하면 어떻게 확인하나요? 당연히 사전의 중간부터 찾아보기 시작하는 것이 아니라, 어떤 목적을 가지고 앞이나 뒤를 바라봅니다.

마찬가지로 예를 들어 0~10000의 값 범위에서 작은 것부터 큰 것까지 균등하게 분포된 100개의 요소가 있는 배열에서 5를 찾으려면 자연스럽게 작은 배열의 첨자를 먼저 고려하여 검색을 시작합니다. .

위의 분석은 사실 이진탐색을 개선한 보간탐색의 아이디어이다.

기본 아이디어:

검색 방법은 검색할 키워드 키를 조회 테이블의 최대 레코드와 최소 레코드의 키워드와 비교하는 것의 핵심입니다. 계산 공식:
$key - $arr[$low]
——————————-
$arr[$high] - $arr[$low]

코드:

<?php
//插值查找(前提是数组必须是有序数组) 事件复杂度 O(logn)
//但对于数组长度比较大,关键字分布又是比较均匀的来说,插值查找的效率比折半查找的效率高
$i = 0;    //存储对比的次数
//@param 待查找数组
//@param 待搜索的数字
function insertsearch($arr,$num){
    $count = count($arr);
    $lower = 0;
    $high = $count - 1;
    global $i;
    while($lower <= $high){
        $i ++; //计数器
        if($arr[$lower] == $num){
            return $lower;
        }
        if($arr[$high] == $num){
            return $high;
        }
        // 折半查找 : $middle = intval(($lower + $high) / 2);
        $middle = intval($lower + ($num - $arr[$lower]) / ($arr[$high] - $arr[$lower]) * ($high - $lower)); 
        if($num < $arr[$middle]){
            $high = $middle - 1;
        }else if($num > $arr[$middle]){
            $lower = $middle + 1;
        }else{
            return $middle;
        }
    }
    return -1;
}
$arr = array(0,1,16,24,35,47,59,62,73,88,99);
$pos = insertsearch($arr,62);
print($pos);
echo "<br>";
echo $i;

요약:

시간 복잡도 관점에서 보면 O(logn)이지만 순서가 지정된 목록의 경우 더 길며 키워드 분포는 상대적으로 균일합니다. 조회 테이블의 경우 보간 검색 알고리즘의 평균 성능이 이진 검색의 성능보다 훨씬 좋습니다. 반대로 배열 내 분포가 {0, 1, 2, 2000, 2001,. . . 999998, 999999} 보간 검색을 사용하는 이러한 종류의 매우 고르지 않은 데이터는 그다지 적합한 선택이 아닐 수 있습니다.

위 내용은 PHP 순차 테이블 검색----보간 검색 내용입니다. 더 많은 관련 내용은 PHP 중국어 홈페이지(www.php.cn)를 참고해주세요!


성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.