数据结构与算法之二分查找

什么是二分查找

  • 二分查找是一种优化的数据查找方法
  • 二分查找是适用前提是:有序数组
  • 二分查找复杂度:O(logn)

二分查找代码实现

  • 有序数组不存在重复数据
//二分查找
function bsearch($array, $size, $value) {
	$low = 0;
	$high = $size - 1;
	
	while ($low <= $high){
		$mid = intval(($low + $high)/2);
		if ($value == $array[$mid]) return $mid;
		else if ($value > $array[$mid]) $low = $mid + 1;
		else $high = $mid - 1;
	}
	return -1;
}

//使用递归实现
function bsearch2($array, $size, $value){
	return tool($array, 0, $size-1, $value);
}
function tool($arr, $low, $high, $value) {
	if ($low > $high) return -1;
	$mid = intval(($low + $high)/2);
	
	if ($value == $array[$mid]) 
		return $mid;
	else if ($value > $array[$mid]) 
		return tool($arr, $low+1, $high, $value);
	else 
		return tool($arr, $low, $mid-1, $value);
}

二分查找变形

  • 有序数组存在重复数据
//查找第一个值等于给定值的元素
function sfirst($array, $size, $value)
{
    $low = 0;
    $high = $size - 1;
    while ($low <= $high) {
        $mid = intval(($low + $high) / 2);
        if ($value > $array[$mid]) {
            $low = $mid + 1;
        } else if ($value < $array[$mid]) {
            $high = $mid - 1;
        } else {
            if (0 == $mid || $array[$mid - 1] != $value) return $mid;
            else $high = $mid - 1;
        }
    }
    return -1;
}

//查找最后一个值等于给定值的元素
function last($array, $size, $value)
{
    $low = 0;
    $high = $size - 1;
    while ($low <= $high) {
        $mid = intval(($low + $high) / 2);
        if ($value > $array[$mid]) {
            $low = $mid + 1;
        } else if ($value < $array[$mid]) {
            $high = $mid - 1;
        } else {
            if ($size - 1 == $mid || $array[$mid + 1] != $value) return $mid;
            else $low = $mid + 1;
        }
    }

    return -1;
}

//查找第一个大于等于给定值的元素
function sfirst_($array, $size, $value)
{
    $low = 0;
    $high = $size - 1;
    while ($low <= $high) {
        $mid = intval(($low + $high) / 2);
        if ($value > $array[$mid]) {
            $low = $mid + 1;
        } else {
            if (0 == $mid || $array[$mid - 1] < $value) {
                return $mid;
            } else {
                $high = $mid - 1;
            }
        }
    }

    return -1;
}

//查找最后一个小于等于给定值的元素
function last_($array, $size, $value)
{
    $low = 0;
    $high = $size - 1;
    while ($low <= $high) {
        $mid = intval(($low + $high) / 2);
        if ($value < $array[$mid]) {
            $high = $mid - 1;
        } else {
            if ($size - 1 == $mid || $array[$mid + 1] > $value) return $mid;
            else $low = $mid + 1;
        }
    }
}

其他

  • 针对数组来说,可以运用二分查找方式获取指定的值
  • 对于链表来说,同样有优化的查询方法,在有序链表上层添加多层索引,查询时通过上层索引层层向下,快速定位到目标,这种数据结构叫跳表
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值