二分查找(折半查找)

二分查找(折半查找):
从有序序列中找到给出的要查询的数字。
原理是:首先把一个有序序列中间位置的值与要查找的数比较,若相等则找到了有序序列中的此数;否则比较两者的大小,若前者大,则把有序序列的中间位置以前的元素都去掉,余下的元素组成一个新的有序数列继续上一步的操作,直到找到为止;若后者大,则把有序数列中间位置以后的元素都去掉,余下的元素组成一个新的有序数列,继续第一步的操作,直到找到为止。若所有元素都与要找的数不同,则说明该有序序列中没有要找的数。
优点是比较次数少,查找速度快,平均性能好;其缺点是要求待查表为有序表,且插入删除困难。因此,折半查找方法适用于不经常变动而查找频繁的有序列表。
具体过程可以参照下图:

这里写图片描述

根据以上概念写出代码:

#include<stdio.h>int *binsearch(int *arr, int n, int se)
{int left = 0;int right = n;int mid = 0;while (left <= right)       //防止有些数没有机会与所输入的数比较{mid = left + (right - left) / 2;//防止left 与 right 直接相加后溢出//这儿可以用右移运算符代替 /2 ,因为右移效率高//mid = (left & right) + ((right ^ left)>>1);与上一句代码的作用相同,(left & right):作用是把left与right对应位相同的数加起来除2, ((right ^ left)>>1):right ^ left):把left和right对应位不同时的数加起来,>>1:把加起来的这个数除2。综合所得:这句代码的作用是把left和right加起来除2.if (arr[mid] == se){return arr + mid;}else if (se > arr[mid]){left = mid + 1;}else{right = mid - 1;}}return NULL;   //当找到数时,返回其在数组中的下标,未找到时返回负值
}int main()
{int num[11] = { 1, 3, 5, 7, 9, 11, 13, 15, 17, 19, 21 };  //折半查找的数组必须是有序的int key = 0;int *ret = 0;int size = sizeof(num) / sizeof(num[0]);printf("Please enter a number:");scanf("%d", &key);ret = binsearch(num, size, key);printf("%d\n", *ret);system("pause");return 0;
}

再次声明,待查找的序列必须是有序的才行。
程序中需要注意的几个点是:
1、自定义二分查找函数的返回值
怎样设定返回值可以合的反应出找到了该数,并能确定他在有序序列中的位置,或者未找到该数。我用的方法是返回下标。因为我的有序序列是一个数组,所以当找到了该数时返回该数在数组中的下标即可确定该数在数组中的位置,没有找到也很好处理,因为下标都是非负数,所以若是没有找到,则返回一个负数。
2、while的循环条件

这里写图片描述

3、mid的值不要写成 mid = (left +right ) / 2 ;因为可能在left 与 right 相加的过程中发生溢出

因为left 和 right都是int型,是有范围的(32字节),若是left 和 right都是很大的数,接近于int能表示的最大数字,那么两数相加后就会溢出,因此用一个较为保险的方法—-mid = left + (right-left)/ 2.

以上使用数组下标的方式实现的,现在介绍一种用指针实现的方法。

int *binsearch(int *start, int count, int num)
{int *end = start + count - 1;while (start <= end){int *mid = start + (end - start) / 2;if (*mid == num){return mid;}else if (*mid>num){if (*(mid - 1) > *(mid + 1)){start = mid + 1;}else{end = mid - 1;}}else{if (*(mid - 1) > *(mid + 1)){end = mid - 1;}else{start = mid + 1;}}}return NULL;
}

这个代码是不考虑数组的排序方式的,不管数组是从大到小排序还是从小到大排序都能解决。

很简单的几十行代码就完成了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/526209.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C语言实现用星号在屏幕上打印菱形

很多人第一感觉肯定都是&#xff1a;很简单啊&#xff0c;不就是多写几个printf 语句嘛 像这样&#xff1a; #define _CRT_SECURE_NO_WARNINGS 1#include<stdio.h>int main() {printf(" *\n");printf(" ***\n");printf(" *****\n");p…

多维数组元素的下标引用与指针访问

在这儿值介绍二维数组元素的访问方式&#xff0c;三维、四维……的与之类似。 先用一维数组的相关知识来抛砖 #define _CRT_SECURE_NO_WARNINGS 1 #include<stdio.h>//1、一维数组元素的下标引用与指针访问 int main() {int arr[10] { 1, 2, 3, 4, 5, 6, 7, 8, 9, 0 }…

scanf()函数错误把输入缓存里的回车作为一次字符输入

有时我们会需要连续的从标准输入中多次读入数据时&#xff0c;那么就可能存在连续使用多次scanf()函数的情况。然而在连续使用scanf()函数时可能会出现一些难以预料&#xff0c;不易发现的坑 #include <stdio.h>int main() {char str[20] { 0 };char c 0;scanf("…

C语言注释与C++注释的相互转换

做此项目的经历主要的收获是熟悉了状态机这一方法的使用&#xff0c;还有就是对每实现一个功能就尽量封装一个函数这一概念把握的更为精到。 状态机&#xff1a;关于状态机的一个极度确切的描述是它是一个有向图形&#xff0c;由一组节点和一组相应的转移函数组成。状态机通过…

运算符优先级与结合性

在编程中经常会有遇到运算符的地方&#xff0c;而每个运算符都有不同的优先级&#xff0c;错误的使用运算符&#xff0c;可能会发生难以预料而又不易发现的错误&#xff0c;因此我们需要掌握运算符的优先级。 优先级的序号越小&#xff0c;优先级越高。

switch()语句块的出口:break;

switch()语句块里的case:相当于一个开关&#xff0c;只要满足case的条件&#xff0c;开关就打开&#xff0c;从而执行case语句块的内容&#xff0c;而break就相当于一个出口&#xff0c;只要碰到了break&#xff0c;就跳出switch语句块&#xff0c;否则就继续执行下一条语句&am…

函数声明

对C缺陷与陷阱里第二章第一节里面的函数声明的一些段落的理解

字符变量存放多个字符

在我们的认知中&#xff0c;不会有字符变量中存放多个字符的情况&#xff08;不要说\n、\t等&#xff0c;这些本来就被C语言标准定义为是一个字符&#xff09;&#xff0c;因为一个字符变量就只占1个字节空间&#xff0c;只能存放一个字符&#xff0c;但往下看&#xff0c;你会…

不同类型的变量与零值比较的方法

在if()、while()等语句块中&#xff0c;我们经常会遇到变量与零值比较的情况&#xff0c;然而不同的类型的变量与零值的比较方法是有一定标准的&#xff0c;编程时最好遵守这些标准&#xff0c;否则会发生一些难以预料的错误。

测试机器大小端的方法

首先&#xff0c;给出一些大小端相关概念。 大小端的由来&#xff1a; 在乔纳森斯威夫特的著名讽刺小说《格列夫游记》中&#xff0c;小人国内部分裂成Big-endian和Little-endian两派&#xff0c;区别在于一派要求从鸡蛋的大头把鸡蛋打破&#xff0c;另一派要求从鸡蛋的小头把…

main()函数参数

很多人学了很长时间的C语言&#xff0c;可能还不知道mian()函数也是有参数的&#xff0c;甚至会说&#xff0c;main()函数哪来的参数&#xff0c;我从来都没见到过&#xff0c;更没有使用过&#xff0c;然而&#xff0c;不得不说&#xff0c;main()函数确实是有参数的。 main函…

运算符求值顺序

&&与||两个运算符的求值顺序是非常重要的&#xff0c;因为很多判断语句都是基于它们拥有一定的求值顺序才能正确进行的。 eg: if((num1 ! 0) && (num2 / num1 \ > num2 - num1)) 上面的if()判断语句合理的运用了&&运算符&#xff0c;使之不会出现…

CentOS设置

我的CentOS是在VMware Workstation里面装的虚拟机&#xff0c;但操作都是一样的 1、设置系统语言 许多人在安装CentOS系统时&#xff0c;可能在开始阶段选择区域及语言时选择了非中文&#xff0c;而却想在中文环境下使用系统&#xff0c;却苦于不知道如何切换到中文系统&…

printf函数的格式修饰符

本文介绍printf函数的一些特殊的格式控制修饰符 1、域宽修饰符 –> 数字 对所有格式控制符都有效&#xff0c;即便是%%也不例外 宽度修饰符出现在%和格式控制符之间的数字&#xff0c;使数据在固定区域打印。如果待打印的数值不能填满位置&#xff0c;它的左侧会被天上空…

线性表之顺序表与单链表的区别及优缺点

这里比较的是基于C语言实现的顺序表与单链表&#xff0c;与其他语言的实现可能会有差异&#xff0c;但我相信语言是相通的&#xff0c;它们的实现机制应该也差不多。 1、What 什么是顺序表和单链表 ①顺序表&#xff1a; 顺序表是在计算机内存中以数组的形式保存的线性表&a…

前置++与后置++之一道简单的题目引发的思考

引言 昨晚一时兴起&#xff0c;我脑子就问自己下面的代码会输出什么&#xff0c;也不知道我脑子为什么有这个代码模型&#xff0c;只是模糊的有些印象&#xff1a; #include <stdio.h> #include <stdlib.h>int main(int argc,char** argv) {int i3,j;j(i)(i)(i);…

有趣的for循环

#include <stdio.h> int main() {int i 0;for(i 0, printf(" First %d", i);i<10, printf(" Second %d", i);i, printf(" Third %d", i)){printf(" Fourth %d", i);}system("pause");return 0; } 这段代码会…

重载函数编译后的新名字

我们都知道很多语言都支持函数重载&#xff0c;那么编译器是怎么处理编译后它们的命名冲突的呢&#xff1f; 1、先给出几个正确的重载函数&#xff1a; #include <iostream> using namespace std;int Add(int a, int b) {return a b; }double Add(double a, double b)…

乱入的'\0'

看这个题之前&#xff0c;先来回忆一下strlen函数的工作机制&#xff1a; strlen所作的仅仅是一个计数器的工作&#xff0c;它从内存的某个位置&#xff08;可以是字符串开头&#xff0c;中间某个位置&#xff0c;甚至是某个不确定的内存区域&#xff09;开始扫描&#xff0c;…

初识C++之剖析const与#define

1、 编译器处理方式不同   #define是一个宏定义命令&#xff0c;它是在预处理阶段就进行了替换&#xff1b;   const修饰的是一个编译时常量&#xff0c;它是在编译阶段处理的。 2、 类型和安全检查不同   #define定义的标识符仅仅是产生文本替换&#xff0c;不管内容…