二分查找(折半查找)算法教学

二分查找(折半查找)算法教学

使用二分查找的好处

在这里我用日常生活举例.比如询问一台电脑的价格,它的价格区间在**(1, 10000)**之间.我们的第一反映肯定是询问价格是超于5000还是低于5000,这样可以排除一半的错误答案,再以此类推,不断砍半.最终会以小于14次折半找到正确答案(2^14^=16384).比1元,2元,3元这种每次加一的询问不知道好多少(现实应该不会遇到这种人).

二分查找在代码中的逻辑(运算过程)

在数组中实现二分查找必须要能满足一个条件--数组是有序数组 ,像我们的价格一样,从1到10000之间是有序的. 我们假设有这样一个数组 像在日常中询问价格一样,我们需要知道它的价格区间,在这里体现为下标的区间.最左边的下标是0 ,那么右边的下标怎么计算呢?一个个算不成? 肯定不是,这里我给大家推荐一种方法:函数sizeof

sizeof(arr)算出的是数组的大小,也就是一共有多少个字节.int类型是4个字节,这个数组10个元素,那就是40个字节.用数组的总大小40个字节除以数组一个元素的大小4个字节就能知道数组的元素为10个 这下子我们也就知道了最右边的下标为10-1=9(下标是由0开始的,而元素个数是由1开始的). 然后跟上面的举例一样,我们要知道它的中间值,也就是 然后我们每次要将我们要查找的数字k(假设k=6),跟数组的中间值arr[mid]进行比较.当mid=4时,arr[mid]=5,5<6.这个时候我们发现一个比较是无法解决问题的,我们要添加循环.在我们进行第二轮查找时,因为arr[mid]<k,所以我们要改变left的值,将left改为mid+1.right不变. 若arr[mid]>k,我们要改变right的值,将right改为mid-1.left不变. 遇到arr[mid]=k,跳出循环. 即`

c 复制代码
    if (k < arr[mid])
		right = mid - 1;
	else if (k > arr[mid])
		left = mid + 1;
	else
		break;

因为在循环中mid的值一直会改变,所以将mid = (left + right) / 2写到循环里面. 循环的判断条件是left<=right.(要判断物品的价格一定要有个区间,就算这个区间只有一个数)

二分查找实现在主函数内

c 复制代码
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
int main() {
	int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
	//            0,1,2,3,4,5,6,7,8,9
	int sz = sizeof(arr) / sizeof(arr[0]);
	int left = 0, right = sz - 1;
	
	int k = 0, mid =0;
	scanf("%d", &k);
	while (left <= right) {
		mid = (left + right) / 2;
		if (k < arr[mid])
			right = mid - 1;
		else if (k > arr[mid])
			left = mid + 1;
		else
			break;
	}
	if (left <= right)
		printf("找到了,下标是%d\n", mid);
	else
		printf("找不到\n");
	return 0;
}

在自定义函数实现二分查找函数

c 复制代码
int find(int arr[], int sz, int k) {
	int left = 0, right = sz - 1,mid=0;
	while (left <= right) {
		mid = (left + right) / 2;
		if (k < arr[mid])
			right = mid - 1;
		else if (k > arr[mid])
			left = mid + 1;
		else
			return mid;
	}
	return -1;
}
int main() {
	int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
	//            0,1,2,3,4,5,6,7,8,9
	int sz = sizeof(arr) / sizeof(arr[0]);
	
	
	int k = 0;
	scanf("%d", &k);

	int ret = find(arr, sz, k);
	if (ret==-1)
		printf("找不到\n");
	else
		printf("找到了,下标是%d\n", ret);
	return 0;
}
c 复制代码
在这里插入代码片

##改进和易错之处 与我们要寻找的元素k相互比较的是数组中的元素,不是中间值mid 2.mid在每次循环中的改变可以写成

c 复制代码
mid = left + (right-left)/2;

这样修改的会让我们的mid更加"耐打".*当我们的left和right都大于int类型最大值的一半或者两者相加大于大于最大值时会导致栈溢出,使得到的mid数据缺失.*这样子可以修改避免二者直接相加.

复制代码
相关推荐
Microsoft Word29 分钟前
c++基础语法
开发语言·c++·算法
天才在此42 分钟前
汽车加油行驶问题-动态规划算法(已在洛谷AC)
算法·动态规划
莫叫石榴姐2 小时前
数据科学与SQL:组距分组分析 | 区间分布问题
大数据·人工智能·sql·深度学习·算法·机器学习·数据挖掘
茶猫_3 小时前
力扣面试题 - 25 二进制数转字符串
c语言·算法·leetcode·职场和发展
肥猪猪爸5 小时前
使用卡尔曼滤波器估计pybullet中的机器人位置
数据结构·人工智能·python·算法·机器人·卡尔曼滤波·pybullet
readmancynn5 小时前
二分基本实现
数据结构·算法
萝卜兽编程5 小时前
优先级队列
c++·算法
盼海5 小时前
排序算法(四)--快速排序
数据结构·算法·排序算法
一直学习永不止步5 小时前
LeetCode题练习与总结:最长回文串--409
java·数据结构·算法·leetcode·字符串·贪心·哈希表
Rstln6 小时前
【DP】个人练习-Leetcode-2019. The Score of Students Solving Math Expression
算法·leetcode·职场和发展