题目:在一个有序数组中查找具体某个数字n
1、方法一:从前往后依次查找
#include <stdio.h>
int main()
{
int arr[] = { 0,11,22,33,44,55,66,77,88,99 }; //假设的有序数列
//元素位置: 0 1 2 3 4 5 6 7 8 9
int k = 0; //k是要查找的数字
scanf("%d", &k); //输入要查找的数字
int i = 0;
int sz = sizeof(arr) / sizeof(arr[0]);
int flag = 0; //判断是否找到的标志
for (i = 0; i < sz; i++)
{
if (k == arr[i])
{
flag = 1;
printf("找到了,下标是:%d\n", i);
break; //如果找到了这个数,便跳出循环
}
}
if (flag == 0) //上面循环从前往后遍历了所有数组的值还没有找到,则说明找不到
printf("找不到\n");
return 0;
}
注:若数组很长,该做法效率低
2、方法二:二分查找(折半查找)
二分查找:通过将区间减半,逐渐缩小查找范围,从而快速完成查找
- (1)首先我们将数组左端元素的下标定义为 left,最右端元素的下标定义为 right。根据左端元素的下标和右端元素的下标求出中间元素的下标 mid
- (2)判断目标数字是在左侧还是右侧【k<arr[mid] — 左侧,k>arr[mid] — 右侧】;若【k==arr[mid]】,该数字即为要查找的数字
- (3)若目标数字在 mid 左侧,把 mid-1 赋给 right;若在右侧,将 mid+1 赋给 left
- (4)通过多次将区间折半,使得所需查找范围越来越小,从而加快查找效率
#include <stdio.h>
int main()
{
int arr[] = { 0,11,22,33,44,55,66,77,88,99 }; //假设的有序数列
//元素位置: 0 1 2 3 4 5 6 7 8 9
int k = 0; //k是要查找的数字
scanf("%d", &k); //输入要查找的数字
int i = 0;
int sz = sizeof(arr) / sizeof(arr[0]);
int left = 0;
int right = sz - 1;
int flag = 0;
while (left<=right) // = 不能忘掉
{
//int mid = (left + right) / 2;【可能导致整型越界问题】
int mid = left + (right - left) / 2; //找到中间数
if (arr[mid] < k) //这个数比中间数大
{
left = mid + 1; //则在右半查找
}
else if (arr[mid] > k) //这个数比中间数小
{
right = mid - 1; //则在左半查找
}
else
{
printf("找到了,下标是:%d\n", mid);
flag = 1;
break; //找到该数便跳出循环
}
}
if (flag == 0)
printf("找不到\n"); //上述循环查找完还没找到,说明此数不在给定的有序数列中
return 0;
}
注:
- (1)折半查找(二分查找),前提必须是数组有序
- (2)第14行:while (left<=right) 中的 = 不能忘记加,因为可能出现两边同时向中间移动,到最后中间只剩一个元素,左右下标都指向该元素的情况
- (3)第16行:int mid = (left + right) / 2; 该方法可行但还可以改进。因为如果考虑数组很长,left 和 right 都接近int型的最大值的话,两数加起来就超越了int型的规定范围,导致越界,这样的数除以2就不是他们两个的平均值了。故优化为【int mid = left + (right - left) / 2;】这样不会出现越界问题。