[ 更换 ]
热门城市
北京上海广州深圳成都杭州南京武汉天津西安重庆青岛沈阳长沙大连厦门无锡福州济南宁波昆明苏州郑州长春合肥南昌哈尔滨常州烟台南宁温州石家庄太原珠海南通扬州贵阳东莞徐州大庆佛山威海洛阳淮安呼和浩特镇江潍坊桂林中山临沂咸阳包头嘉兴惠州泉州三亚赣州九江金华泰安榆林许昌新乡舟山慈溪南阳聊城海口东营淄博漳州保定沧州丹东宜兴绍兴唐山湖州揭阳江阴营口衡阳郴州鄂尔多斯泰州义乌汕头宜昌大同鞍山湘潭盐城马鞍山襄樊长治日照常熟安庆吉林乌鲁木齐兰州秦皇岛肇庆西宁介休滨州台州廊坊邢台株洲德阳绵阳双流平顶山龙岩银川芜湖晋江连云港张家港锦州岳阳长沙县济宁邯郸江门齐齐哈尔昆山柳州绍兴县运城齐河衢州太仓张家口湛江眉山常德盘锦枣庄资阳宜宾赤峰余姚清远蚌埠宁德德州宝鸡牡丹江阜阳莆田诸暨黄石吉安延安拉萨海宁通辽黄山长乐安阳增城桐乡上虞辽阳遵义韶关泸州南平滁州温岭南充景德镇抚顺乌海荆门阳江曲靖邵阳宿迁荆州焦作丹阳丽水延吉茂名梅州渭南葫芦岛娄底滕州上饶富阳内江三明淮南孝感溧阳乐山临汾攀枝花阳泉长葛汉中四平六盘水安顺新余晋城自贡三门峡本溪防城港铁岭随州广安广元天水遂宁萍乡西双版纳绥化鹤壁湘西松原阜新酒泉张家界黔西南保山昭通河池来宾玉溪梧州鹰潭钦州云浮佳木斯克拉玛依呼伦贝尔贺州通化朝阳百色毕节贵港丽江安康德宏朔州伊犁文山楚雄嘉峪关凉山雅安西藏四川广东河北山西辽宁黑龙江江苏浙江安徽福建江西山东河南湖北湖南海南贵州云南陕西甘肃青海台湾内蒙古广西宁夏香港澳门
培训资讯网 - 为兴趣爱好者提供专业的职业培训资讯知识

越“聪明”的代码,越是“一次性”

大多数程序员,面对复杂问题却能写出精简代码时,总会觉得颇有成就感。但本文作者指出:看似越精简的“聪明”代码,其可读性越低,越是得不偿失。

原文链接:https://www.joshwcomeau.com/career/clever-code-considered-harmful/

未经允许,禁止转载!

作者 | Josh Comeau译者 | 弯月 责编 | 郑丽媛出品 | CSDN(ID:CSDNnews)

不可否认,遇到难题能想到巧妙的解决方案会让人很有成就感。例如,挑战自己使用递归而不是迭代,或者创建优雅的级联抽象层以确保代码永远不会重复,你一定会感到很高兴。

对于这类编程,我最喜欢的一个资源是Project Euler(地址:https://projecteuler.net/)。

Project Euler 是一个基于高等数学的挑战库,旨在通过软件解决问题,要求程序在 2004 年左右的硬件上运行时间不超过 1 分钟。这意味着,你不能靠蛮力解决问题,必须想出一个更聪明的解决方案。

下面是一个例子:

越“聪明”的代码,越是“一次性”

问题 58:螺旋质数

从 1 开始,按如下方式逆时针旋转,形成一个边长为 7 的正方形螺旋。

有趣的是,奇数的平方数都位于右下对角线;更有趣的是,位于两条对角线上的 13 个数字中有 8 个是质数,总体占比为:8/13=62%。

如果在上面的螺旋之上再加一层,就会形成一个边长为 9 的正方形螺旋。如果继续这个过程,那么当正方形螺线的边长为多少时,两条对角线上的质数百分比将首次低于 10%?

解决这个问题后,你还可以查看其他人共享的解决方案。你会感到很惊讶:哇!居然有人能想出如此简洁、如此聪明的解决方案?!

下面就是一个用户使用 Haskell 编写的解决方法:

pe58 = nwherea p q = scanl (+) p $ iterate (+ 8) qb = [[x,y,z] | (x,(y,z)) <- zip (a 3 10) $ zip (a 5 12) (a 7 14)]c = zip (scanl1 (+) . map (length . filter isPrime) $ b) (iterate (+ 4) 5)[(n,_)] = take 1 $ dropWhile (\(_,(a,b)) -> 10*a > b) $ zip [3,5..] c

下面是一个用户使用 J 语言编写的代码:

gt;:-:>:1 i.~0.1>}.(>:i.$n)%~+/\n=. isprime +/\1,4#2*>:i.14000

显然,用如此少的代码解决如此困难的问题需要大量技巧。如果我能够写出这样的解决方案,我会为自己感到高兴,但我的解决方案总是又长又不优雅。

不过,日常工作中一般不会出现这样的代码——这些代码只是为了娱乐,编写这样的代码多数是为了凸显自己的聪明、吓唬其他人或锻炼大脑。

上述这类问题的代码,你永远也不会阅读第二次,因为它们都是一次性的,我们也不会交给其他人来维护,这与我们日常编写代码的环境不同。

越“聪明”的代码,越是“一次性”

实习生测试

在谈论日常编写的代码时,我会按照下面这个标准来要求:初级开发人员、或者说刚开始职业生涯的人,是否能够理解这段代码?

所以对于共享的代码库而言,好代码就是简单的代码,不需要任何花里胡哨,便于向新手解释基本概念的代码。

例如几年前,我曾接到一个任务,是审查包含以下函数的拉取请求:

const extractDataFromResponse = (response) => {const [Component, props] = response;const resultsEntries = Object.entries({ Component, props });const assignIfValueTruthy = (o, [k, v]) => (v? { ...o, [k]: v }: o);return resultsEntries.reduce(assignIfValueTruthy, {});}

我给出的建议是,按照下面这样重写:

const extractDataFromResponse = (response) => {const [Component, props] = response;const output = {};if (Component) {output.Component = Component;}if (props) {output.props = props;}return output;}

平心而论,原始版本确实有很多优点:

▶ 代码量更少(只有 4 个语句,而第二个版本中有 7 个语句);

▶ 没有重复(在第二个版本中,两个 if 语句的作用基本相同);

▶ 没有修改任何变量;

▶ 更具可扩展性:重写后可轻松处理 10 个字段,而不止 2 个。

但是,原始版本更难理解——我不得不浪费大量脑细胞来理解这些代码究竟在干什么,对于许多初级开发人员而言,应该也无法很好地理解。

在我看来,函数式版本的可读性成本太高了,得不偿失。

为什么可读性如此重要?

为了说明我认为可读性是好代码的一个重要属性,下面我们来看看流行的开源库 lodash。

越“聪明”的代码,越是“一次性”

lodash 是一个非常流行的库,每周仅通过 NPM 的下载量就超过 2600 万次,Github 上的 Star 数更是超过了 4.2 万个。然而,有一个很奇怪的地方是:这个库未解决的问题始终不会超过 10 个。

尽管我们一直都提倡 Inbox Zero(收件箱归零),但热门项目的问题永远不会归零。但 lodash 的未解决问题经常为零,并且多年来一直如此。

有个主要原因可能是主作者 John-David Dalton 是一位热情的维护者,他花费了大量时间对问题进行分类。但我觉得,无论是任何人,就算是超人,都无法做到让如此热门的一个库的问题数归零——直到在多年前,我在播客上听到 John-David Dalton 谈论管理这类项目的关键是,如何鼓励很多人为代码库做贡献。

他的方法之一是将代码保持在非常基础的水平,使用简单、基本的结构,确保贡献者无论经验深浅都能理解代码并为其做出贡献。我记得,John-David Dalton 提到他们更喜欢 if/else,而不是三元运算符,因为很少有人有使用三元运算符的经验。换句话来说,如果你的目标是保持代码简单易懂,那么用三元运算符的表达方式就是有害的。

如果你正在构建开源工具,那么记住这一点很重要;如果你正在与其他人一起开发生产代码库,那么这一点就尤为重要,特别是当队友的经验不如你时。

bug数与代码长度

你听过下面这句话吗?

“更少的代码意味着更少的错误隐藏空间。”

这句话的意思是,代码要越短越好,因为这样更容易发现错误,而每多敲一个字符,都会增加出错的可能性。

对于拼写错误这类的 bug,上述论点固然没错,但拼写错误往往很容易被发现和修复。真正麻烦的 bug 是由代码过于复杂而引发的,开发人员遇到这类 bug 就好像拿到烫手山芋,恨不得立即甩锅给别人。

在调试问题时,你必须全神贯注,仔细理解每一行代码。当你创建一个抽象来减少重复(所谓“DRY 原则”,Don"t Repeat Yourself,不要重复你自己)时,无形中就添加了一个间接层,理解起来也就多了一重障碍。思维模型越难解释每一个边缘情况和可能的状态,就越难诊断出问题的关键所在。

“每个人都知道调试比写程序要难一倍。所以,如果你在编写代码时机关算尽,那么将来要如何调试呢?”

—— Brian Kernighan,《The Elements of Programming Style》

抽象的成本

如果目标是降低复杂性,而抽象增加了复杂性,那么我们是否应该完全废除抽象呢?并不是。因为抽象无处不在:循环是抽象,函数是抽象,编程语言本身就是对机器代码的抽象,而机器代码本身则是晶体管的抽象。

所以说,一切都是抽象。关键在于,我们需要权衡抽象的成本与收益。

假设我们正在构建一个 React 应用,我们有一个列表,其中包含 100 个需要渲染的东西。我们可以选择复制/粘贴同一个 JSX 100 次,也可以将其映射到一个数组,这样只需编写一次 JSX。在这种情况下,“复杂”的解决方案是完全值得的,因为底层的复杂性众所周知,而替代方案维护起来会很麻烦。

在构建产品时,我们总是需要做出这样的权衡决定。我个人的看法是,我们在权衡利弊时应该考虑年轻队友,想一想我们为他们增加了多少复杂性?这样值得吗?

不可避免的复杂性

有时,有些代码本身就很复杂,因为现实世界就很复杂,我们的软件必须对其建模。我们无法确保总是能写出初级工程师可以轻松解析和贡献的代码。有时是因为业务逻辑真的很棘手,有时是因为我们不得不使用理解起来有难度的 API 等等。

我认为,一个最好的解决这类问题的方法是尝试隔离复杂性。在简单与复杂之间设置清晰的界限。不要让复杂性渗透到周围区域。

John-David Dalton 在 lodash 中做到了这一点。他在播客采访中表示,绝大多数 lodash 代码都简单易懂,他们将复杂的部分集中到了一起,组成核心代码,用于处理难题。这意味着,大多数贡献者不必在意这些复杂的代码,因为这种复杂性不会散布到整个应用中。

如果你的应用架构合理,复杂的问题都集中在一个地方处理,那么应用的绝大部分代码都可以保持简单。如果初级工程师需要处理那个复杂的核心,该怎么办?那么就只能找一个更资深的人来指导他们完成了。

冒名顶替综合症

去年的 React Rally 上,Chantastic 发表了一篇我最喜欢的演讲,题目是“Hot Garbage; Clean Code is Dead”。我的收获之一是明白每个人都患有冒名顶替综合症,我们总是会想尽办法证明自己,如果我编写了一个超级聪明且难以理解的函数,同事就会觉得我很聪明。

然而,我逐渐意识到,任何人都可以编写看似复杂的代码,而最难的部分实际上是用简单的代码来解决复杂的问题。如果你真的掌握了这种本领,没有人会质疑你的能力。

相关内容

怎么学?如何干?西藏7市(地)委书记在西藏日报发表文章

启航新征程 开创新局面为全面建设社会主义现代化新拉萨努力奋斗自治区党委常委、拉萨市委书记 普布顿珠党的二十大大笔擘画坚持以中国式现代化全面推进中华民族伟大复兴的宏伟蓝图,区党委十届三次全会全面铺开建设美丽幸福西藏、共圆伟大复兴梦想的壮阔实践···

2023年全区智慧旅游专业人才培训班圆满结业

6月2日,自治区旅游发展厅主办的全区智慧旅游专业人才培训班在拉萨圆满结业。培训共历时三天,累计完成全区旅游行政管理部门工作人员及涉旅企业专业技术人员培训100人。本次培训是区旅发厅深入推进学习贯彻习近平新时代中国特色社会主义思想主题教育重要···

徐汇萨迦共同举办大美西藏首届口腔学习班

随着现代医学的发展,口腔健康越来越受到人们的关注,为提高西藏地区口腔技术水平和服务质量,近日,徐汇区牙病防治所和萨迦县中心医院远程连线共同举办首届口腔学习班开班仪式。徐汇区卫生健康委副主任胡强,上海援藏干部、萨迦县委常务副书记、常务副县长沈···

自觉抵制“一对一”“一对多”等学科类培训!哈尔滨中小学生暑期预警来了

17日,记者从哈尔滨市各区教育局获悉,2023年暑假将至,南岗区、道里区、香坊区教育局向家长发出预警,自觉抵制违规培训,各区义务教育阶段学科类培训机构已经全部注销,暑假期间以任何形式开展的学科类培训均属于违规培训。家长要自觉抵制任何机构或个···

哈尔滨市道里区企投局举办“招商大讲堂”专题培训

黑龙江网讯(记者 王惠婷)10月24日,哈尔滨市道里区企投局组织开展“招商大讲堂”专题培训班,全区各招商专班负责同志及业务骨干参加培训。本次培训从实际需求出发,紧紧围绕当前招商工作中的热点、难点,对“什么是招商引资”“道里区的产业结构和主导···

团黑龙江省委举办全省青年文明号青年岗位能手学习宣传贯彻党的二十大精神培训交流会

中国青年报客户端讯(李海涛)为深入学习贯彻党的二十大精神,充分发挥青年文明号、青年岗位能手示范引领作用,在全省职业青年中掀起学习党的二十大精神热潮,11月22日,团黑龙江省委举办全省青年文明号青年岗位能手学习宣传贯彻党的二十大精神培训交流会···

辽宁葫芦岛举办外贸政策培训会推动外贸保稳增量

辽宁省葫芦岛市外贸政策培训会4月3日举办。 辽宁省贸促会供图中新网葫芦岛4月3日电 (李晛)辽宁省葫芦岛市外贸政策培训会4月3日举办。本次活动由辽宁省贸促会支持、葫芦岛市商务局主办,葫芦岛市贸促会、葫芦岛海关、中国出口信用保险辽宁分公司和辽···

山西运城:严查无证校外培训机构 查封9家警告2家

新华社太原8月5日电(记者王飞航)记者从山西省运城市政府了解到,运城市教育局近日联合市公安局等多家单位,对中心城区无证校外培训机构进行了一次突击检查,共检查了13家校外培训机构,查封9家,警告2家,发放整改通知书4份。今年7月,运城市教育局···

山西开展培训筑牢森林“防火墙”

山西新闻网3月30日讯(记者 卢奕如)今日,记者从山西省应急管理厅获悉,全省举办森林草原防灭火业务培训,邀请专家以视频会议形式,围绕森林扑火指挥实操、森林草原火灾防控经验做法、火灾现场各级各类指挥员具体操作中遇到的问题等内容进行授课。培训内···

校外培训机构治理工作取得进展 山西停办近1300所

资料图:小学生排队等待进入校园。中新社记者 刘文华 摄中新网5月11日电 据教育部网站消息,按照校外培训机构专项治理工作整体安排,教育部、民政部、国家市场监管总局启动了校外培训机构治理专项督查工作。5月9日至10日,督查组率先在北京市开展华···

山西综改区举办省技术创新中心申报培训

  8月18日,山西综改区科技金融部举办2023年度省技术创新中心申报培训会,来自区内企业、科研院所及有关单位代表160余人参加了培训。  山西省技术创新中心是以产业前沿引领技术和关键共性技术研发为核心的产业技术创新平台,承担着为区域和产业···

山西汾阳医院开展健康教育与控烟知识培训

来源:【吕梁日报-吕梁新闻网】本报讯 (记者 刘少伟) 5月18日,在“世界无烟日”到来之际,山西汾阳医院组织开展健康教育与控烟知识培训。近年来,山西汾阳医院全面落实健康中国战略,根据国家卫健委《关于2011年起全国医疗卫生系统全面禁烟的决···

山西省文物局年度田野考古技术培训班开班

10月10日,山西省文物局在运城闻喜上郭城址、邱家庄墓群举办2023年度田野考古技术培训班开班仪式。该次培训为期三个月,通过理论和实践两部分教学,旨在推进山西考古工作高质量发展,提升考古业务人员专业技术水平。本次培训由山西省考古研究院和山西···

最低每课时9元!全省学科类校外培训课时长和收费标准出台

近日,山西省发改委、省教育厅下发《关于中小学学科类校外培训收费标准及有关事项的通知》,明确全省中小学学科类校外培训收费标准,从12月17日起执行。《通知》对全省线上线下学科类校外培训基准收费标准和浮动幅度制定了科学标准。其中,义务教育阶段线···

山西天镇 阳光职业培训学校培养乡村“新农人”乡村振兴添动能

(记者 贺文生) 山西天镇县阳光职业培训学校紧紧围绕乡村振兴战略,按照“政府引导、农民自愿、立足产业、突出重点”的原则,创新高素质农民技能培训方式方法,采取以“授人以渔”的方式,让人才振兴成为助推农业农村现代化的内生动力,以高素质农民引领现···

山西:建立全过程 全链条 无缝隙安全培训制度

黄河新闻网讯(记者杨江涛)日前,山西省应急管理厅下发了《山西省安全培训管理暂行办法》(以下简称《办法》)。我省将进一步抓好安全生产这个基本盘、基本面,推动全省安全培训工作制度化、规范化、科学化,促进安全培训工作高质量发展。山西省应急管理厅厅···

山西:艺考培训机构纳入全国监管平台管理

央广网太原10月6日消息(记者郎麒) 日前,山西省教育厅、省发改委、省公安厅等部门联合制定《加强面向高中阶段学生艺考培训规范管理工作方案》,针对艺考培训的突出特点和实际情况,全面规范艺考培训行为,将艺考培训机构统一纳入全国校外教育培训监管与···

太平财险阳泉中支开展消防安全教育和有限空间作业培训

为强化员工安全意识,进一步提升员工消防和有限空间突发事件应急处理能力,9月14日,太平财险阳泉中支邀请北京市卫民安消防教育咨询中心山西分中心讲师向全体员工开展了一次消防安全教育和有限空间作业课程培训。按照防消结合、预防为主的原则,本次讲座通···

山西省数字化转型贯标试点工作宣贯培训会在太原举行

  10月20日消息,山西省数字化转型贯标试点工作宣贯培训会在太原举行。省工信厅介绍,作为国家数字化转型贯标试点省份,试点启动后将引导企业加快数字化转型,助力制造业高端化、智能化、绿色化发展。  今年,工信部启动数字化转型贯标试点工作,我省···

山西马兰花创业培训讲师大赛收官 太原市获多个奖项

山西新闻网8月31日讯(记者 冯耿姝)8月29日,山西省第四届马兰花创业培训讲师大赛圆满收官,太原市代表队在比赛中分获多个二、三等奖和优秀奖。本届大赛以“启迪创新思维·激发创业梦想”为主题,全省共有56名教师晋级复赛,其中,太原市有7名选手···