·使用besearch函数的前提(一些废话)

首先让我们先亮出二分法的定义:

https://baike.baidu.com/item/二分法/1364267

以及二分法实现的方法:

https://blog.csdn.net/sufeiboy/article/details/54401257

这些应该是使用二分查找前需要了解的知识,综上我们可以得出:使用besearch前应该先将目标数组进行一定规律的排序,事实上大部分时候我们会使用库中自带的qsort函数进行排序。

·besearch函数的函数原型解析

(资料源于网络)

void *bsearch(const void *key, const void *base, size_t num, size_t size, int (*cmp)(const void *, const void *));

解释一下参数

key: 为要查找的元素的地址

base: 指向进行查找的数组的开始地址

num: 为想要查找的范围,即在多少个数中进行二分查找

size:数组中每个元素的大小,一般用sizeof()表示

cmp:比较两个元素的函数,定义比较规则。这里我们可以类比qsort,首先对于这样一个cmp:
int cmp(const void *p1,const void *p2);
其中p1始终指向的是key的地址,而p2指向的是besearch所查找的数组里的传递过来的元素
其次我们知道qsort对于cmp函数的比较返回值的处理是当返回值为正则交换两个元素,而对于bsearch则为当返回值为0则认为查找到了目标元素。当返回值为正数则besearch会向后进行二分查找,返回值为负数将会向前进行二分查找。 简单的记忆方法便是:升序数组返回p1-p2,降序数组返回p2-p1(这里建议看完使用实例后再看)

关于besearch的返回值: besearch在找到元素后将返回数组中满足cmp条件的这个数的地址,但是这个地址是void *类型的,如果我们不加以类型转换就无法使用。而如果没有找到对应元素则会返回NULL指针,我们可以通过判断返回值的类型得到元素是否在数组中存在。
关于besearch的陷阱: 当数组中存在多个匹配元素时,besearch不能保证返回的指针一定指向某一个特定的元素,此时besearch只能用于证明数组内是否含有特定元素

·bserach的使用实例

#include <stdio.h>
#include <stdlib.h> int cmp(const void *p1,const void *p2)
{
int a=*(int *)p1,b=*(int *)p2;//为了清楚说明进行临时转化保存
return a-b;//仅当a=b时返回0,说明查找到了
}
int main(int argc, char const *argv[])
{
int arry[100];
for(int i=1;i<=100;i++) arry[i-1]=i;
int *ans;//用于接受查找的结果
int key;
scanf("%d",&key);
ans=(int *)bsearch(&key,arry,100,sizeof(int),cmp);//记得对结果进行转换
if (ans!=NULL)
printf("The key:%d exists.The pointer is %X.",key,ans);
else
printf("The key doesn't exists\n");
return 0;
}

四次运行实例:

输入:
13
输出:
The key:13 exists.The pointer is 61FD74.
输入:
60
输出:
The key:60 exists.The pointer is 61FD9C.
输入:
0
输出:
The key doesn't exists
输入:
101
输出
The key doesn't exists

·bserach的使用实例

我们知道,上述的数组是一个升序数组,那么考虑一下情况,如果arry是一个 100 99 98…… 2 1的降序数组此时我们该如何编写cmp函数
其实很简单,根据cmp返回值对besearch的影响,我只需把return语句重写

原本:
int cmp(const void *p1,const void *p2)
{
int a=*(int *)p1,b=*(int *)p2;
return a-b;
}
现在
int cmp(const void *p1,const void *p2)
{
int a=*(int *)p1,b=*(int *)p2;
return b-a;
}

是不是很像qsort的处理方式呢
更进一步的,如果我们要寻找这样一个元素它恰好是key的三倍,那么我们可以这么写:

int cmp(const void *p1,const void *p2)
{
int a=*(int *)p1,b=*(int *)p2;
return 3*a-b;//这里默认arry进行升序排列
}

至此,关于besearch的基本部分便已陈述完毕,实际上我们可以通过besearch进行更复杂的查找,正如网上很多的关于qsort对结构体进行一级甚至多级排序,besearch也同样可以。
本文在此结束,但读者可以自行去探索。

深入理解CPP与C中bsearch函数的用法的更多相关文章

  1. Oracle 中 CONTAINS 函数的用法

    Oracle 中 CONTAINS 函数的用法 1. 查询住址在北京的学生 SELECT student_id,student_name FROM students WHERE CONTAINS( a ...

  2. matlab中patch函数的用法

    http://blog.sina.com.cn/s/blog_707b64550100z1nz.html matlab中patch函数的用法——emily (2011-11-18 17:20:33) ...

  3. mysql中INSTR函数的用法

    mysql中INSTR函数的用法 INSTR(字段名, 字符串) 这个函数返回字符串在某一个字段的内容中的位置, 没有找到字符串返回0,否则返回位置(从1开始) SELECT * FROM tblTo ...

  4. (转)解析PHP中ob_start()函数的用法

    本篇文章是对PHP中ob_start()函数的用法进行了详细的分析介绍,需要的朋友参考下     ob_start()函数用于打开缓冲区,比如header()函数之前如果就有输出,包括回车/空格/换行 ...

  5. Delphi中 StrToIntDef函数的用法

    Delphi中 StrToIntDef函数的用法:比如我要判断一个文本框里输入的字符串能不能转换为integer类型,如果能,则返回转换后的整型数据,如果不能,则返回整数0,那么我就可以用strtoi ...

  6. Python中int()函数的用法浅析

      int()是Python的一个内部函数 Python系统帮助里面是这么说的 >>> help(int)  Help on class int in module __builti ...

  7. matlab中repmat函数的用法(堆叠矩阵)

    matlab中repmat函数的用法 B = repmat(A,m,n) B = repmat(A,[m n]) B = repmat(A,[m n p...]) 这是一个处理大矩阵且内容有重复时使用 ...

  8. Matlab中imfilter()函数的用法

    Matlab中imfilter()函数的用法 功能:对任意类型数组或多维图像进行滤波.用法:B = imfilter(A,H) B = imfilter(A,H,option1,option2,... ...

  9. Oracle trunc()函数,decode()函数,substr函数,GREATEST函数,java中substring函数的用法

    --Oracle trunc()函数的用法/**************日期********************/1.select trunc(sysdate) from dual --2013- ...

随机推荐

  1. file 多次上传附件功能完善

    之前解决了一个页面中的单个附件上传问题,使用的是 id 定位.但是一个页面中,可能存在多个附件上传的地方,这时候如果继续使用 id,会出问题. 我依旧会上传一个附件.附件链接地址: https://f ...

  2. Node.js目录

    [相关学习] npm入门教程 [基础] (1) 初识Node.js (2) 开发环境和调试工具 (3) commonJs 规范 (4) node 概念(global.process进程.调试) (5) ...

  3. Koa与Node.js开发实战(3)——Nunjucks模板在Koa中的应用(视频演示)

    技术架构: ​ 在Koa中应用Nunjucks,需要先把Nunjucks集成为符合Koa规格的中间件(Middleware),从本质上来讲,集成后的中间件的作用是给上下文对象绑定一个render(vi ...

  4. 通过<meta>标签指定IE的文档模式实现CSS3兼容

    今天发现之前做好的一个页面在IE中打开显示的效果不正常,本地和服务器上显示的是两种不同的样式. 经过确认文档内容和CSS都是一样的. 通过IE F12(开发人员工具)发现不正常的样式 浏览器文档模式自 ...

  5. 程序到CPU的路径

    汇编 源码->编译->CPU C/C++ 源码->编译->机器码->系统(执行)->CPU Java/.NET 源码->编译->J字节码->虚拟机 ...

  6. 20164305徐广皓 - Exp1 PC平台逆向破解(5)M

    1.逆向及Bof基础实践说明        1.1实践目标 实践对象:pwn1的linux可执行文件 实践目的:使程序执行另一个代码(ShellCode) 实践内容: 手工修改可执行文件,改变程序执行 ...

  7. arguments.callee.caller

    1.Arguments Arguments是一个类似数组但不是数组的对象,说它类似数组是因为其具有数组一样的访问性质及方式,可以由arguments[n]来访问对应的单个参数的值,并拥有数组长度属性l ...

  8. 【项目】Selenium和pymongo复习

    import pymongo client = pymongo.MongoClient(host='localhost',port=27017) db = client.test collection ...

  9. Linux 使用 arp-scan 检查是否存在IP地址冲突

    如果前期没有做好IP地址规划,即使有IP地址统一不小心也会犯错!推荐服务器IP地址使用要登记明细,上次机房批量部署服务器,就将已再用的IP又分配给另一台服务器,还好对业务没有造成大的影响. 那么在给服 ...

  10. Java基础11-List;Set;Map

    作业解析: remove(int index); //删除指定位置的元素 List list = new ArrayList(); list.add("s1"); list.add ...