为什么要拒绝嵌套代码?

news2024/11/15 1:34:40

linux 内核的最早作者 linus torvalds 在 linux 内核样式指南 第一节中提到:

if you need more than 3 levels of indentation, you’re screwed anyway, and should fix your program.

如果你需要超过3级的缩进,你无论如何都搞砸了,应该修复你的程序。

什么是嵌套

嵌套代码是指您向函数中添加更多的内部块。一般认为,每一个左大括号都为函数增加了一个深度。下面介绍一些例子:

int calculate(int bottom, int top)
{
  return bottom + top;
}

此函数的嵌套深度为 1.

在此基础上再增加一个 if 语句:

int calculate(int bottom, int top)
{
  if (top > bottom)
  {
    return bottom + top;
  }
  else
  {
    return 0;
  }
}

该函数的嵌套深度增加为 2.

如果再增加一个循环:

int calculate(int bottom, int top)
{
  if (top > bottom)
  {
    int sum = 0;
    for (int number = bottom; number <= top; number++)
    {
      sum += number;
    }
    return sum;
  }
  else
  {
    return 0;
  }
}

此时,calculate 函数有三个左大括号,那么我们认为该函数的深度为 3.

许多优秀的工程师都认为,三层嵌套是优秀编程范式的最大深度。

如果超过了三层,那么随之而来的复杂程度将会成倍增加,复杂的函数逻辑将会使阅读和修改变得困难甚至毫无可能。

尤其是在千万行级别的大型工程中,这种毫无制约的嵌套将会是一个定时炸弹!

那么如何重构代码,使其去嵌套化呢?

去嵌套化

我们有两种方式重构代码,使其嵌套深度小于等于 3提取和反转。这两者结合可以达到我们的目的。

提取是指将复杂函数中的特定部分提取出来,使其成为一个新的函数。

反转是指反转 ifelse 中的代码块,使其能够提前 return ,继而去除 else 语句,减小嵌套深度。

首先给出一个四层深度的实例代码,此代码在上述 for 循环中增加了一个 if 语句:

int calculate(int bottom, int top)
{ // level one
  if (top > bottom)
  { // level two
    int sum = 0;
    for (int number = bottom; number <= top; number++)
    { // level three
      if (number % 2 == 0)
      { // level four
        sum += number;
      }
    }
    return sum;
  }
  else
  {
    return 0;
  }
}

提取

for 循环内部部分提取出来,形成一个单独的函数 filterNumber.

int filterNumber(int number)
{
  if (number % 2 == 0)
  {
    return number;
  }
  return 0;
}

int calculate(int bottom, int top)
{ // level one
  if (top > bottom)
  { // level two
    int sum = 0;
    for (int number = bottom; number <= top; number++)
    { // level three
      sum += filterNumber(number);
    }
    return sum;
  }
  else
  { // level two
    return 0;
  }
}

此时的函数深度为 3.

反转

反转的思路就是,将一些错误判定条件(提前 return,或者抛出异常)放到真正的业务代码前面,这样就可以省略了许多的 else 语句,避免陷入更深的嵌套中。

接下来我们反转 ifelse 中的语句,并将 if 中的条件取反:

int filterNumber(int number)
{
  if (number % 2 == 0)
  {
    return number;
  }
  return 0;
}

int calculate(int bottom, int top)
{ // level one
  if (top <= bottom)
  { // level two
    return 0;
  }
  else
  { // level two
    int sum = 0;
    for (int number = bottom; number <= top; number++)
    { // level three
      sum += filterNumber(number);
    }
    return sum;
  }
}

实际上容易发现,我们并不需要这个 else 语句块,因为如果满足那个会导致 提前return的条件,那么语句将会按照正常顺序执行。所以去掉 else

int filterNumber(int number)
{
  if (number % 2 == 0)
  {
    return number;
  }
  return 0;
}

int calculate(int bottom, int top)
{ // level one
  if (top <= bottom)
  { // level two
    return 0;
  }
  int sum = 0;
  for (int number = bottom; number <= top; number++)
  { // level two
    sum += filterNumber(number);
  }
  return sum;
}

此时可以发现,该函数的嵌套深度已经缩小为 2,代码逻辑更加清晰,易读性大大增加。

结语

本文只是举了一个十分简单的例子,在实践中遇到的代码将会更加复杂。

其实万变不离其宗,就是这两种最实用的方法:提取和反转。多次,灵活的运用这两种方法,将会使你的程序更加清晰明了,符合国际主流代码风范。


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/196767.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

ABBYY15免费照片识别文字识别软件

照片识别文字的软件有哪些&#xff1f;在日常的工作的时候&#xff0c;小伙伴们是不是经常会借助拍摄照片记录一些比较重要的事情或者是一些比较重要的笔记&#xff0c;等回头空闲出来的时候在去整理&#xff0c;那么这时候&#xff0c;如果我们运用好图片识别文字这项技术&…

线性代数篇

主线为花书第二章-线性代数&#xff0c;但其上面一些表述属实费解&#xff0c;于是参考B站3Blue1Brown线性代数和B站同济子豪兄的视频讲解。 先放一句3B1B的话共勉&#xff0c;伙计们不要被数学公式吓到&#xff0c;慢慢钻研&#xff0c;慢慢推肯定能学懂。线性代数这一部分相信…

cannot redeclare block-scoped variable

使用 Vue3.0 的时在模块类中声明方法的时候抛出了如下的异常:cannot redeclare block-scoped variable解决办法检查是否集成了 Vetur 插件,若存在禁用或卸载即可,该插件Vue3.0 的时候会冲突;Vue3.0 集成如下两款即可:名称: TypeScript Vue Plugin (Volar)ID: Vue.vscode-typesc…

Qt 自定义控件 带UI 不带 UI

一般自定义控件原因 有时Qt 现有控件不能满足我们的开发需求&#xff0c;这时候就需要我们进行自定义控件的使用&#xff0c;自定义控件,这大大提高了设计UI的通用性,程序利用&#xff0c;封装&#xff1b; Part1 easy one 继续 Lab 自定义Label控件&#xff1a;文本太长省略…

机械设备管理系统如何帮助企业做好成本核算管理?

随着多元化市场经济的深入发展&#xff0c;机械设备制造企业面临的竞争压力也越来越大&#xff0c;企业要想在激烈的市场环境中生存下去&#xff0c;就得不断提高竞争力。企业提高自身竞争能力最为重要一个途径便是不断提高企业的成本核算及管控能力&#xff0c;降低企业经营成…

java使用导出百万级别数据

用过POI的人都知道&#xff0c;在POI以前的版本中并不支持大数据量的处理&#xff0c;如果数据量过多还会常报OOM错误&#xff0c; 这时候调整JVM的配置参数也不是一个好对策&#xff08;注&#xff1a;jdk在32位系统中支持的内存不能超过2个G&#xff0c;而在64位中没有限制&a…

栈、堆、全局区/静态存储区、常量区、代码段、到底是什么?

一、程序运行内存分布图 我们知道一个由我们编写好的程序&#xff0c;运行时&#xff0c;我们的程序中写的代码&#xff0c;定义的变量&#xff0c;写的函数、for 循环等等&#xff0c;这些运行时都分布在内存中的哪里吗&#xff1f; 一下是一个程序运行时 内存的各个区域的分…

C语言字符串指针(指向字符串的指针)详解

C语言中没有特定的字符串类型&#xff0c;我们通常是将字符串放在一个字符数组中&#xff0c;这里演示一下&#xff1a;#include<stdio.h> #include<string.h>intmain(){ char str[]"http://csdn.net"; int len strlen(str), i; //直接输出字符串 printf…

一套完全开源,支持多租户,界面配置单点的后端框架JVS,赶紧收藏

今天推荐的这个项目是「JVS数据全家桶中的 JVS微服务框架」—— 是一个免费开源的中后台模版&#xff0c;使用了最新的 vuespring cloud 主流技术开发&#xff0c;开箱即用的中后台前端解决方案&#xff0c;可以直接商用&#xff0c;并且这个脚手架上做了很多商业上的扩展&…

day19 二叉树 | 654、最大二叉树 617、合并二叉树 700、二叉搜索树中的搜索 98、验证二叉搜索树

题目 654、最大二叉树 给定一个不含重复元素的整数数组。一个以此数组构建的最大二叉树定义如下&#xff1a; 二叉树的根是数组中的最大元素。左子树是通过数组中最大值左边部分构造出的最大二叉树。右子树是通过数组中最大值右边部分构造出的最大二叉树。 通过给定的数组构…

数据中心转型利润中心:数据如何赋能零售行业营销升级?

通过大数据来提升营销效果已经是零售行业不同业态都绕不过去的话题。以往&#xff0c;市场团队在进行投放时需要依靠营销团队和销售团队的经验来判断&#xff0c;这样的营销策略较为粗放&#xff0c;对用户的感知也不精准&#xff0c;一般都是在海量投放后被动的等待营销反馈。…

C++代码优化(2):条款5~12

"然后自然老去吧&#xff0c;别再依依惜别了"条款5:了解C默默编写并调用了哪些函数(1)小试牛刀地回顾C编译器为类提供的默认函数很多人知道是有6个的。默认构造、拷贝构造、赋值重载、析构函数、重载取地址运算符&#xff0c;但是在C11更新后&#xff0c;又为类对象增…

软件测试项目实战,我们拿到项目第一步应该怎么做【附过程文档】

对于从事软件研发的组织来说&#xff0c;工作类型至少包括项目管理、产品设计、编码、测试、质量保证和软件配置管理&#xff0c;以及其它人员&#xff0c;如文档编制人员和美工人员/系统硬件管理人员等。根据职能需要&#xff0c;可以以半独立方式进行部门和项目的矩阵管理&am…

Let’s Encrypt共建安全的互联网

导读最近关于沃通和 StartCom 这两家 CA 公司的消息让人们再次关注到了网络隐私和安全的问题。随着 Mozilla、苹果和谷歌对这两家 CA 公司处罚落定&#xff0c;很多使用这两家 CA 所签发证书的网站纷纷寻求新的证书签发商。这里面固然有不少可信赖的 CA 公司可以提供服务&#…

UniRx之基本语法格式

前言 想要更好的去学习UniRx&#xff0c;我们最好是先理解UniRx 的语法格式。就像我们去学习英语或者汉语一样&#xff0c;理解了基本的语法规则后&#xff0c;再学其他的就信手拈来了。 语法示例 下面我们来看一个最简单的示例&#xff0c; Observable.EveryUpdate().Where…

Git安装,配置及Gitee项目代码pull到本地

一、Git安装从git官网下载&#xff1a;https://git-scm.com/downloads选择适合的版本进入下载&#xff0c;然后傻瓜式安装直到结束。检查安装是否成功&#xff1a;进入任意文件夹点击右键选择Git Bash Here进行基本配置输入如下命令&#xff1a;git --version就会显示当前安装的…

【Windows】ip地址修改器v5.0.5.4

简介 IP地址修改器&#xff0c;一款能够快速的切换IP地址&#xff0c;在几个不同的固定IP之间进行切换&#xff0c;手动输太麻烦&#xff0c;所以可以用到这款IP地址修改器&#xff01; 下载 ip地址修改器v5.0.5.4 软件介绍 程序主要原理还是利用了WMI的Win32_NetworkAda…

二叉搜索树,平衡二叉树,红黑树,B树,B+树

文章目录二叉树&#xff08;BT&#xff09;1. 满二叉树2. 完全二叉树二叉搜索树&#xff08;BST&#xff09;平衡二叉搜索树&#xff08;AVL&#xff09;1. 定义2. 如何保持平衡——旋转红黑树&#xff08;RBTree&#xff09;1.定义2.红黑规则3.插入规则B树1.定义2.在磁盘系统中…

云存储、云计算与分布式存储、分布式计算是一回事吗?

随着互联网的蓬勃兴起&#xff0c;大数据、人工智能、物联网、云计算与云存储等这些专业词汇在大众视野内出现的频率越来越高&#xff0c;再加上近几年分布式技术异军突起&#xff0c;更使得分布式存储、分布式计算等成为热词。然而&#xff0c;很多人对这些名词都一知半解&…

HTML5+CSS3(七)-全面详解(学习总结---从入门到深化)

目录 字体属性 color font-size font-weight font-style font-family 学习效果反馈 背景属性一 background-color属性 background-image属性 background-repeat属性 学习效果反馈 背景属性二 background-size属性 background-position属性 background-attachment属性…