哨兵节点:思想简单,效果很棒的编程算法

所谓的哨兵,就是一个标志,一个与查找目标对象一样的操作对象。

别人的经验,我们的阶梯!

今天和同事一起调代码,定位到一处很耗时的地方。

在某个线程中,同步周期需要保证在​2​​​毫秒(如果耗时不到​​2​​​毫秒,那么就让剩下的时间进行​​sleep​​)。

但是在调用一个模块的内部函数时,时不时的就飘到了​​3~5​​毫秒,时间抖动毫无保证。

后来仔细分析了一下被调用的函数,发现是在查找链表中某个目标节点时,由于目标节点的不确定性,导致耗时飘来飘去。

后来想到是否可以用"哨兵"的思路来解决问题,于是就试了一下,果然有效。

特分享于此,使用​​2​​段代码来看一下代码执行效率的提升。

普通的算法

所谓的哨兵,就是一个标志,一个与查找目标对象一样的操作对象。

以前有一本书中举过这样的一个例子:

假如有​​10000​​​个纸箱,每个箱子里面都有一张纸条,纸条上写有​​1 ~ 10000​​这些数字,数字不会重复。

现在:别人给了一个随机的数字,我们需要在这​​10000​​个箱子里找到与这个数字相同的纸条,找到之后退出操作。

面对这个问题,最直觉的想法就是:从头开始,遍历这​​10000​​个箱子,检查其中的纸条上数字是否与目标相同。

因为纸箱里的纸条不是按照顺序排列的,所以只能从头开始遍历;

大概就是下面这个样子:

复制int lookfor_num = xxx;

for (int i = 0; i < 10000; ++i)

{

if (box[i] == lookfor_num)

{

printf("找到了!箱子编号是:%d n", i);

break;

}

}1.2.3.4.5.6.7.8.9.从上面这段示意性代码中可以看出,在​​for​​​循环中主要有​​2​​个比较指令:

比较箱子的编号 i 是否到了最后一个箱子;比较箱子里的纸条上数字,是否与要查找的目标数字相同;

为了便于量化问题,我们写一个测试代码,打印出​​for​​循环的时间消耗。

为了便于客观比较,在测试代码中:

循环次数设置为 500000 万次;箱子里纸条上的数字按顺序存放,不影响讨论问题的本质;查找的数字设置为一个中间值 500000;

测试文件:​​loop1.c​​。

复制#include <unistd.h>

#include <stdio.h>

#include <stdlib.h>

#include <time.h>

#include <sys/time.h>

#define LOOP_NUM 1000000

int main(int argc, char *argv[])

{

long data[LOOP_NUM];

long rand_num = 500000;

struct timeval tv1, tv2;

for (long i = 0; i < LOOP_NUM; ++i)

{

data[i] = i;

}

gettimeofday(&tv1, 0);

for (long i = 0; i < LOOP_NUM; ++i)

{

if (data[i] == rand_num)

{

printf("hit rand_num. i = %ld n", i);

break;

}

}

gettimeofday(&tv2, 0);

long us1 = tv1.tv_sec * 1000000 + tv1.tv_usec;

long us2 = tv2.tv_sec * 1000000 + tv2.tv_usec;

printf("time elapse: %ld n", us2 - us1);

return 0;

}1.2.3.4.5.6.7.8.9.10.11.12.13.14.15.16.17.18.19.20.21.22.23.24.25.26.27.28.29.30.31.32.编译:​​gcc loop1.c -o loop1​​。

执行:

耗时大概在​​1350 ~ 1380​​微秒左右。

哨兵算法

哨兵算法的主要思想就是:降低在​​for​​循环中的比较操作。

因为纸箱的数量是有限的,上面的代码中,在还没有找到目标数字之前,需要对纸箱的序号进行检查:以免超过了最大的纸箱。

我们可以在最后额外添加一个纸箱,并且在其中存放我们查找的目标数字,额外添加的这个纸箱就叫做​​哨兵​​!

这样的话,在​​for​​循环中,就不需要检查当前这个纸箱序号是否超过了最大的纸箱。

因为:我们在哨兵纸箱中放了被查找的那个数字,所以是一定能够找到目标数字的:

要么是在前面的纸箱中, 要么是在哨兵纸箱中!

因此,在​​for​​循环中,就只需要比较纸条上的数字,而不用比较纸箱的序号是否达到最后一个了。

当找到目标数字之后,唯一要多做的步骤是:检查这个箱子是否为哨兵纸箱。

如果是哨兵纸箱:说明前面的纸箱中没有查找到目标数字。

如果不是哨兵纸箱:说明在前面的纸箱中查找到了目标数字。

测试代码​​loop2.c​​:

复制#include <unistd.h>

#include <stdio.h>

#include <stdlib.h>

#include <time.h>

#include <sys/time.h>

#define LOOP_NUM 1000000

int main(int argc, char *argv[])

{

long data[LOOP_NUM + 1]; // add another room

long rand_num = 500000;

struct timeval tv1, tv2;

for (long i = 0; i < LOOP_NUM; ++i)

{

data[i] = i;

}

data[LOOP_NUM] = rand_num; // add a sentinel

gettimeofday(&tv1, 0);

long i = 0;

while (1)

{

if (data[i] == rand_num)

{

if (i != LOOP_NUM)

{

printf("hit rand_num. i = %ld n", i);

break;

}

}

++i;

}

gettimeofday(&tv2, 0);

long us1 = tv1.tv_sec * 1000000 + tv1.tv_usec;

long us2 = tv2.tv_sec * 1000000 + tv2.tv_usec;

printf("time elapse: %ld n", us2 - us1);

return 0;

}1.2.3.4.5.6.7.8.9.10.11.12.13.14.15.16.17.18.19.20.21.22.23.24.25.26.27.28.29.30.31.32.33.34.35.36.37.38.39.编译:​​gcc loop2.c -o loop2​​。

执行:

耗时大概在​​960 ~ 990​​微秒之间。

小结

这篇短文仅仅是用​​for​​循环来讨论哨兵的编程思想。

在其它的一些编程场景中,应用的机会还是挺多的,也能够非常显著的提升代码的执行效率。

责任编辑:姜华来源: IOT物联网小镇

原文地址:https://www.toutiao.com/article/7105935019419582991/

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


所以很多人都会选择将这些生活琐事来交给智能化产品,在众多产品中,扫拖机器人所给我们带来的便利性最强,扫地、拖地全都一气呵成,不需要人为过多干预,不过目前大多数扫拖机器人对于双手解放得不彻底。而石头作为
“昨天的经历都将成为明天的力量。” 将绝大部分精力都奉献给璃月港的刻晴,可以说是《原神》中的高人气角色了,虽然是常驻角色,并且对于普通玩家来说……刻师傅别刮了……不过作为开服就存在的角色,还有独特的剑法
最近,全球知名的通信产业盛会MWC 2024(2024世界移动通信大会)正式召开,其中,联发科以“连接AI宇宙”(Connecting the AI-verse)为主题,为大众展示出一系列在AI和移动通信技术等领域的最新突破,吸引了大量行业
今年上半年有很多值得关注的机型,其中华为最新的影像旗舰华为P70 Art也自然受到了业界不少的关注目光,目前关于这款机型的轮廓图已经在网上曝光。
目前,2024世界移动通信大会(MWC)正在西班牙巴塞罗那举行,值得一提的是,此次大会参展中国厂商非常多,包括华为、中兴、小米、荣耀等等多家厂商均在其列。
就在去年,真我推出了11 Pro+,用一个2亿像素传感器和zoom变焦功能,开启了中端手机影像的长焦大战,而后友商才姗姗来迟的跟进了2亿像素传感器。
【手机之家新闻】一年一度的MWC已经于当地时间2月26日在巴塞罗那正式开展,在本次MWC2024上全球各大厂商齐聚一堂,展出自家最新的技术与产品,其中中兴就参展本次MWC2024,并且展出了诸多面向企业端的产品,而旗下的
近日,联发科在MWC 2024(2024 世界移动通信大会)上展出了一系列令人瞩目的AI和移动通信技术突破,以“连接AI宇宙”(Connecting the AI-verse)的展厅吸引了无数业界精英和媒体的目光。特别是其现场的生成式AI技术
虽然目前国内已经有不少厂商入局折叠屏产品,但是努比亚却迟迟没有入局。不过在近日举办的MWC 2024展会上,努比亚发布了自家首款折叠屏手机——努比亚Flip,预计国内很快也会上市。
MWC 2024正在西班牙巴塞罗那举办,和往年一样,荣耀这次依旧携众多新产品、新技术参会。荣耀Magic6 Pro、荣耀Magic V2 RSR保时捷设计的机型在海外正式发布,并且还展示了魔法大模型、任意门等诸多新技术。
MWC 2024正在西班牙巴塞罗那如火如荼地举行,其中小米也参加了今年的大会,在会上发布了在国内大受欢迎的小尺寸旗舰——小米14。值得一提的是,高通公司CEO安蒙甚至亲临发布会现场为这款机型助阵。
《原神》是一直以来在机圈深受欢迎的游戏,在充满幻想的提瓦特大陆上,你可以邂逅不少性格迥异、能力独特的伙伴。而一加Ace系列一直就拥有非常强烈的电竞属性,也是畅玩《原神》的热门机型,而在本月,一加Ace 3将推
有不少网友发现,今年新机的发布时间相对于往年大幅提前,很多厂商在春节之前密集发布了自己最新的中高端机型,给人一种年后没什么新机可发了的感觉。不过魅族全新的大杯机型——魅族21 PRO非常值得期待,魅族科技也
2022年7月,小米12S Ultra正式发布,这款产品率先将1英寸大底主摄引入到移动影像领域,同时凭借鲜明的徕卡影调给人留下深刻的印象,同时这款产品也被视为了影像旗舰地位的机型。如果从那时算起,到现在差不多已经快过
随着智能手机的日益普及和智能化进程的加速,智能穿戴设备成为了人们关注的焦点。各大智能手机厂商纷纷进军智能穿戴市场,试图在这一新兴领域抢占先机。
早在去年秋天,HyperOS操作系统发布的时候,小米便勾勒出了“人车家全生态”的美好蓝图,而在这其中,小米的多终端统一战略是核心,目前已经有不少小米产品预装或者接受到了HyperOS操作系统的推送,在过去几个月的时
今年雷军将把更多的精力放在小米汽车上,所以接下来的手机业务将由刚刚兼任小米品牌总经理卢伟冰接管。同时雷军也在微博上表示小米2024年开年旗舰——小米14 Ultra即将在近期发布,并且将有卢伟冰进行讲解。另外,卢
新的一年有龙则灵,有愿必达。自1月19日起,荣耀加码“新年荣耀,一起成龙”年货节,在全国荣耀线下门店上线了“新年许愿处”、“龙运当头”等趣味活动,吸引大批消费者到店打卡许愿,戴龙头迎好运。与此同时,为了回
小米在官网微博中已经透露了关于小米14 Ultra信息,所以新机上市应该不会太晚。根据德国莱茵的官方消息,目前小米14 Ultra(型号为24030PN60G)获得了莱茵无频闪认证,表明这款手机可以有效减轻屏幕给用户带来的视觉疲
2月22日,上海广播电视台与华为举办鸿蒙合作签约仪式,宣布其官方客户端看看新闻APP将基于HarmonyOS NEXT鸿蒙星河版启动鸿蒙原生应用开发,为用户提供更加极致的新闻资讯服务体验。此次合作标志着上海广播电视台成为全国