二分查找(折半查找)

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

这里写图片描述

根据以上概念写出代码:

#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…

分号不是个“随便的人”

分号是个很不起眼的东西&#xff0c;但在C语言编程里面&#xff0c;它的作用不可忽视&#xff0c;多写与漏写都会造成不可预料的错误&#xff0c;下面就列举一些常见的错误。 1、多谢了分号 if或while语句块内如果只有一条语句&#xff0c;那么可以省略大括号的书写&#xff…

函数声明

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

字符变量存放多个字符

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

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

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

悬挂“else”

else匹配规则&#xff1a;与它上面的距离它最近的且在同一级语句块的一个未与else匹配的if语句匹配 初接触C语言的同志可能会犯下面这种错误 if(x 0if(y 0){printf("这里有两个if语句\n");} else {printf("只有一个else语句&#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;使之不会出现…

跳出循环的三种方式

break、continue、goto三个关键字都可以用于跳出循环&#xff0c;但三者跳出循环的情况却有所不同 braek&#xff1a;跳出本层循环&#xff0c;执行本层循环下面的语句。 continue&#xff1a;终止本次循环&#xff0c;进入下一次循环。 goto&#xff1a;调到标号所在位置&a…

C语言实现静态顺序表

1、用于存放声明的头文件 #ifndef __SEP__LIST__ #define __SEP__LIST__#define _CRT_SECURE_NO_WARNINGS 1 #include <stdio.h> #include <assert.h>#define MAX_SIZE 10 //使顺序表中可以插入的数据是可以很方便调整的 typedef int DataType;typede…

C语言实现动态顺序表

1、用于存放声明的头文件 #pragma once //避免重复引入头文件&#xff0c;作用于预编译指令几乎相同#define _CRT_SECURE_NO_WARNINGS 1 #include <stdio.h> #include <assert.h> #include <malloc.h>typedef int DataType; typedef struct SepList_d {Dat…

CentOS设置

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

VS Code无法远程ubuntu

①windows下cmd中执行命令&#xff1a; ssh-keygen -t rsa -C "xxxxxx.com" ②ubuntu 下bash 命令 ssh-keygen -t rsa -C "xxxxxx.com" ③cat id_rsa.pub >> ~/.ssh/authorized_keys ④ C:\Users\Administrator\AppData\Roaming\Code\User\set…

C语言实现单链表

1、用于存放声明的头文件 #define _CRT_SECURE_NO_WARNINGS #pragma once #include <stdio.h> #include <assert.h> #include <malloc.h>typedef int DataType;typedef struct SListNode {DataType data; //链表中存放的数据struct SLis…