二分查找的相关算法题

文章讲述了在有序数组和旋转数组中进行查找的高效方法,包括二分查找的应用、处理重复元素的策略,以及如何在特定情况下选择顺序查找。作者还分享了针对有重复元素的数组和无重复元素数组的搜索代码示例。
摘要由CSDN通过智能技术生成

int indexMid = index1;

while(numbers[index1] >= numbers[index2])

{

if(index2 - index1 == 1)

{

indexMid = index2;

break;

}

indexMid = (index1 + index2) / 2;

//如果下标为index1、index2和indexMid指向的三个数字相等,则只能顺序查找

if(numbers[index1] == numbers[index2] && numbers[indexMid] == numbers[index1])

return MinInOrder(numbers , index1 , index2);

if(numbers[indexMid] >= numbers[index1])

index1 = indexMid;

else if(numbers[indexMid] <= numbers[index2])

index2 = indexMid;

}

return numbers[indexMid];

}

//顺序查找

int MinInOrder(int *numbers , int index1 , int index2)

{

int result = numbers[index1];

for(int i = index1 + 1 ; i <= index2 ; ++i)

{

if(result > numbers[i])

result = numbers[i];

}

return result;

}

注意:当两个指针指向的数字及他们中间的数字三者相同的时候,我们无法判断中间的数字是位于前面的字数组还是后面的子数组中,也就无法移动两个指针来缩小查找的范围。此时,我们不得不采用顺序查找的方法。

2 旋转数组中查找某个数字

要求

给定一没有重复元素的旋转数组(它对应的原数组是有序的),求给定元素在旋转数组内的下标(不存在的返回-1)。

例如

有序数组为{0,1,2,4,5,6,7},它的一个旋转数组为{4,5,6,7,0,1,2}。

  • 元素6在旋转数组内,返回2

  • 元素3不在旋转数组内,返回-1

分析

遍历一遍,可以轻松搞定,时间复杂度为O(n),因为是有序数组旋转得到,这样做肯定不是最优解。有序,本能反映用二分查找,举个例子看看特点

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

可以看出中间位置两段起码有一个是有序的(不是左边,就是右边),那么就可以在有序的范围内使用二分查找;如果不再有序范围内,就到另一半去找。

参考代码

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

int search(int A[], int n, int target) {

int beg = 0;

int end = n - 1;

while (beg <= end)

{

int mid = beg + (end - beg) / 2;

if(A[mid] == target)

return mid;

if(A[beg] <= A[mid])

{

if(A[beg] <= target && target < A[mid])

end = mid - 1;

else

beg = mid + 1;

}

else

{

if(A[mid] < target && target <= A[end])

beg = mid + 1;

else

end = mid - 1;

}

}

return -1;

}

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

扩展

上边的有求是没有重复的元素,现在稍微扩展下,可以有重复的元素,其他的要求不变。

思路

大致思路与原来相同,这是需要比较A[beg] 与 A[mid]的关系

  • A[beg]  < A[mid] ————左边有序

  • A[beg]  > A[mid] ————右边有序

  • A[beg]  = A[mid] ————++beg

boolean search(int A[], int n, int target) {

int beg = 0;

int end = n - 1;

while (beg <= end)

{

int mid = beg + (end - beg) / 2;

if(A[mid] == target)

return true;

if(A[beg] < A[mid])

{

if(A[beg] <= target && target < A[mid])

end = mid - 1;

else

beg = mid + 1;

}

else 0if(A[beg] > A[mid])

{

if(A[mid] < target && target <= A[end])

beg = mid + 1;

else

end = mid - 1;

}

else

++beg;

}

return false;

}

3 数字在排序数组中的出现次数

//二分查找,二分查找key第一次出现的位置,二分查找最后一次出现的key

//返回两者相减+1或者找到第一次出现的位置,向后查找

int binarySearchFirstPos(int * iArr, int l, int h, int key)

{

while(l <= h )

{

int mid = (l + h) / 2;

if(iArr[mid] < key)

l = mid +1;

elseif(iArr[mid] > key)

h = mid - 1;

else

{

if(mid == l || iArr[mid - 1] != key)

return mid;

else

h = mid - 1;

}

}

return -1;

}

int binarySearchLastPos(int * iArr, int l, int h, int key)

{

while(l <= h)

{

int mid = (l + h) / 2;

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

新的开始

改变人生,没有什么捷径可言,这条路需要自己亲自去走一走,只有深入思考,不断反思总结,保持学习的热情,一步一步构建自己完整的知识体系,才是最终的制胜之道,也是程序员应该承担的使命。

《系列学习视频》

《系列学习文档》

《我的大厂面试之旅》

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

mg2.imgtp.com/2024/03/13/H4lCoPEF.jpg" />

新的开始

改变人生,没有什么捷径可言,这条路需要自己亲自去走一走,只有深入思考,不断反思总结,保持学习的热情,一步一步构建自己完整的知识体系,才是最终的制胜之道,也是程序员应该承担的使命。

《系列学习视频》
[外链图片转存中…(img-A5PvvUWs-1713500991545)]

《系列学习文档》

[外链图片转存中…(img-RGgnrkjK-1713500991546)]

《我的大厂面试之旅》

[外链图片转存中…(img-WIrCtMyv-1713500991547)]

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 12
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值