百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

“把 if 往上提,for 往下放!”(在if-else语句中,当条件成立时,就执行什么后的语句组)

myzbx 2025-07-09 16:07 3 浏览

很多程序员写代码的时候都会遇到这种情况:一个判断条件到底该放在函数里面还是外面?循环里是不是可以加个 if?这些看起来无关紧要的小选择,实际会影响代码的清晰度、性能。最近,一位热爱简单代码和编程语言的程序员 Alex Kladov 在博客上分享了两条简单但特别实用的经验法则,引发不少开发者的注意。其表示:“把 if 往上提,把 for 往下放(Push Ifs Up, Push Fors Down)”。这会对提升代码质量有哪些作用,我们不妨通过本文一起来看看。

作者 | Alex Kladov 翻译 | 苏宓
出品 | CSDN(ID:CSDNnews)

关于两条相关经验法则的简短说明。


把 If 条件往上推

如果函数内部有 if 条件判断,可以考虑是否能把它移到调用者那里:

// 好的写法fn frobnicate(walrus: Walrus) { ...}// 不好的写法fn frobnicate(walrus: Option<Walrus>) { let walrus = match walrus { Some(it) => it, None => return, }; ...}

就像上面举的例子一样,函数里经常会有“前置条件”的检查——比如你传进来的参数是不是有效、是不是空之类的。很多时候,程序员会在函数内部去检查这些情况,如果不符合就什么都不做。但更好的做法,其实是把这些判断提前,在调用这个函数之前就处理好。这样一来,函数本身就可以更专注地完成它的核心工作,不用管太多额外的事。

为什么要这么做?有两个原因:

首先,如果每个函数都自己检查一次条件,那整个程序里可能会有很多重复判断,既费事又容易出错。把判断提前处理好,可以让代码整体更简洁高效。

其次,if 语句(也就是“如果……就……”)本身就是让程序变复杂的东西。判断多了,程序流程就绕了,Bug 也容易藏在里面。所以与其到处乱插 if,不如集中放在一个地方处理——比如放在主函数里统一决定怎么走,然后把具体的执行任务交给其他小函数去做,这样结构更清晰,出错的概率也更低。

而且,如果所有的判断都集中在一个地方,程序员在看代码的时候也能更容易发现有没有重复判断,或者根本不会被执行的“死代码”。比如:

fn f() { if foo && bar { if foo { } else { } }}fn g() { if foo && bar { h() }}fn h() { if foo { } else { }}

对于 f() 来说,更容易看出无用分支,而分散在 g()h() 的组合里不容易发现。

这里还有一个相关的模式,叫做“dissolving enum”重构。有时候代码会长这样:

enum E { Foo(i32), Bar(String),}fn main() { let e = f(); g(e)}fn f() -> E { if condition { E::Foo(x) } else { E::Bar(y) }}fn g(e: E) { match e { E::Foo(x) => foo(x), E::Bar(y) => bar(y) }}

这里有两处分支,通过往上提,可以发现它们其实是同一个条件被重复了三次(第三次变成了数据结构):

fn main() { if condition { foo(x) } else { bar(y) }}

把 For 循环往下推

这条建议来自“面向数据”的编程思想。简单说就是:少就是少,多就是多。现实中的程序,大多数时候都不是在处理一个东西,而是一大批东西。特别是在那些对性能要求高的关键部分(也叫“热点路径”),几乎总是在同时处理很多个对象——正是因为数量多,这部分代码才会“热”。

所以,一个常见的好习惯是:从一开始就把“批量处理”当成主要方式,而不是只想着一个个地处理。如果真的需要处理单个对象,那就把它当成“批量里的一个特例”来对待。这样做,不光能让程序结构更清晰,还能为后面提升性能打下基础。比如:

// 好的写法frobnicate_batch(walruses)// 不好的写法for walrus in walruses { frobnicate(walrus)}

这样做最主要的好处是性能,尤其在极端情况下非常明显。

如果有一整批数据可以一起处理,就能摊销启动成本,还能灵活调整处理顺序。实际上,你甚至不必按特定顺序处理实体,可以先对所有实体的某个字段统一处理,再处理其他字段,利用向量化或结构化数组技巧。

一个有趣的例子是基于 FFT 的多项式乘法:在一堆点上同时计算多项式,比一个点一个点计算要快得多!

如果 if 往上推,for 循环往下推,两条规则还能组合使用:

// 好的写法if condition { for walrus in walruses { walrus.frobnicate() }} else { for walrus in walruses { walrus.transmogrify() }}// 不好的写法for walrus in walruses { if condition { walrus.frobnicate() } else { walrus.transmogrify() }}

好的写法避免了在循环内部反复判断条件,减少了循环里的分支,提高了性能,甚至可能解锁向量化优化。

这个模式既适用于微观层面,也适用于宏观架构层面,比如 TigerBeetle 的数据平面就是对批量对象操作,以摊销控制平面决策的成本。

虽然性能是“for 循环往下推”的主要动机,但它也能提升表达力。比如早期很成功的 jQuery 就是针对元素集合的操作,抽象向量空间的语言思维往往比逐个坐标方程更清晰。

最后用一句话总结:把 if 条件往上推,把 for 循环往下推!


2025 全球产品经理技术大会

时间:2025 年 8 月 15–16 日

地点:北京·威斯汀酒店

2025 全球产品经理技术大会将汇聚互联网大厂、AI 创业公司、ToB/ToC 实战一线的产品人,围绕产品设计、用户体验、增长运营、智能落地等核心议题,展开 12 大专题分享,洞察趋势、拆解路径、对话未来。

相关推荐

男人的内裤,到底可以穿多久?(男人内裤最多能穿几天)

女生们如果家里有男生可能会发现——他们对内裤很恋旧穿到褪色松垮穿到别有洞天穿到一网情深穿到人间蒸发都仍然...舍不得这位老伙计男生们到底有多热爱旧内裤?有外国媒体曾在街头采访,发现:女士们往往会随...

typeof 与 instanceof 区别(typeof与instanceof区别)

typeof操作符返回一个字符串,表示未经计算的操作数的类型使用方法如下:typeofoperandtypeof(operand)operand表示对象或原始值的表达式,其类型将被返回举个例子...

年纪轻轻病情就已是晚期!你还敢再喝这种饮料吗?

本文作者:谢祥成,浙江大学医学院附属邵逸夫医院肾内科主任医师吴俊男,浙江大学医学院附属邵逸夫医院肾内科主治医师30岁的金先生(化名)是一名才华横溢的设计师。半年前出现视物模糊,起初以为是用眼过度,没有...

typeof 与 instanceof 有什么区别

typeof和instanceof是JavaScript中用于类型检查的两个操作符,但它们的用途和适用场景有显著区别。以下是它们的区别及使用注意事项:1.typeof作用:返回一个变量的基本...

数据结构之顺序表(数据结构顺序表图书管理系统)

线性表定义线性表是n(n≥0)个具有相同特性的数据元素的有限序列。记作:(a1,a2,…,ai-1,ai,ai+1,…,an)线性表相关概念直接前驱元素:ai-1领先于ai,称a...

每一个成熟的人,都需要具备「翻篇」这种能力

“翻篇儿”——仿佛读出这个儿化音,才够表达那种潇洒的感觉是一种人生中非常重要的心理过程和心理技能。人生注定不完美,我们只要活着就会遭遇不愉快的经历,只有及时翻篇儿,才能把更多注意力放在当下,不被过去的...

打通 JAVA 与内核系列之 一 ReentrantLock 锁的实现原理

写JAVA代码的同学都知道,JAVA里的锁有两大类,一类是synchronized锁,一类是concurrent包里的锁(JUC锁)。其中synchronized锁是JAVA语言层面提供的能力,在此不...

韩国吃货主播,美食声控咀嚼音,你是搬运工,好吃到停不下来

刘姐畅谈。Hey,Hongsi。TodaywehaveassortedtoysthatImade。Foryouguysfirst。Itlookscrunchybecause...

黄子韬2019新歌最好的我们完整歌词介绍在哪可以听

最好的我们(TheBestofUs)-黄子韬词:黄子韬曲:黄子韬编曲:DarylK制作人:DarylK助理制作:郭舒文和音:黄子韬电吉他:CalvinC木吉他:雷十一录音室:Kong...

刷一道LeetCode -- 三数之和(三数之和算法)

原题:https://leetcode-cn.com/problems/3sum/给你一个包含n个整数的数组nums,判断nums中是否存在三个元素a,b,c,使得a+b+c...

隐藏在阳光当中的地球刺客(隐藏在阳光下的秘密)

小行星什么时候会撞击地球?这一直是大家比较关注的问题,特别是当大家知道地球上前一任住户是亡于小行星之后,就更加关注这个问题了。图1尤卡坦半岛的陨石坑(NASA)实际上,地球每天都会遭受到一些天体的袭...

安卓手机爱奇艺app中离线视频导出

安卓手机爱奇艺app中离线视频导出:通常我在爱奇艺中发现好的视频,想保存下来,点击离线缓存,缓存好后,在手机上可以查看,但是使用手机连接电脑打开后,发现保存视频的文件夹是空的。1)在手机中爱奇艺文...

50款经典奥斯汀月季,超多图片,抗病好养新手必种的月季

【50款经典奥斯汀月季】大家好,今天来给大家介绍50款经典的奥斯汀月季,奥斯汀是一位伟大的育种家,以他命名的奥斯汀公司也繁育出了数量众多的月季品种。根据木木的种植经验,奥斯汀的月季大多植株长势良...

你也想像J姐一样在梦幻芭比大house里“哭泣”吗?

“6年前我的兜里只揣着400元美金,现如今我已经住上了这上亿豪宅”他是一个我行我素,敢说敢做的一个网红博主他测评过的彩妆都卖断货了他的自创同名品牌深受好评他就是JeffreeStar,你们传说中的J...

VB Do While\Until,Loop循环语句

DoWhile\Until…….Loop循环语句上一节讲了For……Next循环语句,这节讲DoWhile\Until…….Loop循环语句。有人会有疑问,既然有For循环,还要Do循环干什么?它...