重构手法之重新组织函数

2024-09-03 18:38

本文主要是介绍重构手法之重新组织函数,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

重构手法之重新组织函数

在重构的手法中,很大的一部分是对函数进行整理,使函数能够恰当地包装代码(让代码自己说话而不是写更多的注释)。重新组织函数的驱动力,往往都是由于函数过长。因为函数过长就以为着包含了更多属性和逻辑,这样复杂的逻辑和诸多属性(如函数内部的局部变量或者静态变量等)会让代码变得难以维护,需要对其进行重新组织。

提炼函数

在冗长的函数中提炼出精小的函数,让每个短小函数负责的功能简洁,并且让函数名解释函数的用途。这是因为:
+ 如果每个函数的粒度都很小,那么函数被复用的机会就越大,组合就越发灵活
+ 函数粒度越小,覆写的时候也会更加容易
总而言之:易于重用,易于维护

以查询取代临时变量

提炼函数中临时变量是一个让人头疼的问题,我们需要消除某些临时变量。对于那些用于保存求值结果的临时变量,我们可以利用一个方法来取代之。下面是一个例子:

double rate = _whatUsed / _total;
if (rate > 0)return true;
elsereturn false;

利用一个方法来消除double rate可以写成:

/*double rate(){return  _whatUsed / _total;}
*/
if (this->rate())return true;
elsereturn flase;

当我们利用一个查询来替换临时变量,那么在这个类中个的所有方法都可以获取该信息,而不是仅仅只有临时变量原本所在的函数才能获取该信息。这样在我们提取一个新的方法时即可不通过参数传递的方式就可以直接获取该信息。这是一个准则:如果有某种信息,需要在类中不止一个方法中使用,那么该信息应当属于类而不是属于某个函数,在后文中以函数对象替换函数的手法也是基于此原则的。

引入解释性变量

其实写代码的时候写注释只是一种补充手段,我们为了增加代码的可读性应该让代码自己说话而不是尽量多地去注释,过多的注释会在代码中带来背景干扰。引入解释性变量就是基于让代码自己说话的想法引入的。下面是一个例子:

if (plamtform.toUppercase().indexof("MAC") >  0 &&browser.toUppercase().indexof("IE") > 0){//...do something
}

引入解释变量之后如下:

auto isMAC = plamtform.toUppercase().indexof("MAC") > -1;
auto isIE = browser.toUppercase().indexof("IE") > -1;if (isMAC && isIE){//do something....
}

代码可读性大大提高。

分解临时变量

在代码中对一个临时变量进行多次赋值使用,则意味着一个临时变量拥有多重语意,这会在后期维护中带来疑惑,需要分解该临时变量。下面是一个例子:

auto temp =  (_hight + _width) * 2;
std::cout << temp << std::endl;
temp = _hight * _width;
std::cout << temp << std::endl;

分解变量之后:

auto perimeter =  (_hight + _width) * 2;
std::cout << perimeter << std::endl;
auto area = _hight * _width;
std::cout << area << std::endl;

以函数对象替换函数

在进行函数提炼的时候,如果被提炼的函数有许多临时变量,并且提炼出来的子函数会用到这些临时变量。这样可以将这些临时变量作为参数传递到子函数当中,但是如果用到的临时变量很多,那么函数的参数列表就会变得很长。于是我们可以使用函数对象来替换原来的函数。这样,原函数中的临时变量现在就可以变成类的一个属性,可以被所有的方法共享,于是提炼出来的子函数,就可以不依赖临时变量,提取子函数就会变得十分方便,因为不用考虑临时变量的问题。

这篇关于重构手法之重新组织函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1133725

相关文章

hdu1171(母函数或多重背包)

题意:把物品分成两份,使得价值最接近 可以用背包,或者是母函数来解,母函数(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v)(1 + x^v+x^2v+.....+x^num*v) 其中指数为价值,每一项的数目为(该物品数+1)个 代码如下: #include<iostream>#include<algorithm>

C++操作符重载实例(独立函数)

C++操作符重载实例,我们把坐标值CVector的加法进行重载,计算c3=c1+c2时,也就是计算x3=x1+x2,y3=y1+y2,今天我们以独立函数的方式重载操作符+(加号),以下是C++代码: c1802.cpp源代码: D:\YcjWork\CppTour>vim c1802.cpp #include <iostream>using namespace std;/*** 以独立函数

函数式编程思想

我们经常会用到各种各样的编程思想,例如面向过程、面向对象。不过笔者在该博客简单介绍一下函数式编程思想. 如果对函数式编程思想进行概括,就是f(x) = na(x) , y=uf(x)…至于其他的编程思想,可能是y=a(x)+b(x)+c(x)…,也有可能是y=f(x)=f(x)/a + f(x)/b+f(x)/c… 面向过程的指令式编程 面向过程,简单理解就是y=a(x)+b(x)+c(x)

利用matlab bar函数绘制较为复杂的柱状图,并在图中进行适当标注

示例代码和结果如下:小疑问:如何自动选择合适的坐标位置对柱状图的数值大小进行标注?😂 clear; close all;x = 1:3;aa=[28.6321521955954 26.2453660695847 21.69102348512086.93747104431360 6.25442246899816 3.342835958564245.51365061796319 4.87

OpenCV结构分析与形状描述符(11)椭圆拟合函数fitEllipse()的使用

操作系统:ubuntu22.04 OpenCV版本:OpenCV4.9 IDE:Visual Studio Code 编程语言:C++11 算法描述 围绕一组2D点拟合一个椭圆。 该函数计算出一个椭圆,该椭圆在最小二乘意义上最好地拟合一组2D点。它返回一个内切椭圆的旋转矩形。使用了由[90]描述的第一个算法。开发者应该注意,由于数据点靠近包含的 Mat 元素的边界,返回的椭圆/旋转矩形数据

Unity3D 运动之Move函数和translate

CharacterController.Move 移动 function Move (motion : Vector3) : CollisionFlags Description描述 A more complex move function taking absolute movement deltas. 一个更加复杂的运动函数,每次都绝对运动。 Attempts to

✨机器学习笔记(二)—— 线性回归、代价函数、梯度下降

1️⃣线性回归(linear regression) f w , b ( x ) = w x + b f_{w,b}(x) = wx + b fw,b​(x)=wx+b 🎈A linear regression model predicting house prices: 如图是机器学习通过监督学习运用线性回归模型来预测房价的例子,当房屋大小为1250 f e e t 2 feet^

封装MySQL操作时Where条件语句的组织

在对数据库进行封装的过程中,条件语句应该是相对难以处理的,毕竟条件语句太过于多样性。 条件语句大致分为以下几种: 1、单一条件,比如:where id = 1; 2、多个条件,相互间关系统一。比如:where id > 10 and age > 20 and score < 60; 3、多个条件,相互间关系不统一。比如:where (id > 10 OR age > 20) AND sco

JavaSE(十三)——函数式编程(Lambda表达式、方法引用、Stream流)

函数式编程 函数式编程 是 Java 8 引入的一个重要特性,它允许开发者以函数作为一等公民(first-class citizens)的方式编程,即函数可以作为参数传递给其他函数,也可以作为返回值。 这极大地提高了代码的可读性、可维护性和复用性。函数式编程的核心概念包括高阶函数、Lambda 表达式、函数式接口、流(Streams)和 Optional 类等。 函数式编程的核心是Lambda

PHP APC缓存函数使用教程

APC,全称是Alternative PHP Cache,官方翻译叫”可选PHP缓存”。它为我们提供了缓存和优化PHP的中间代码的框架。 APC的缓存分两部分:系统缓存和用户数据缓存。(Linux APC扩展安装) 系统缓存 它是指APC把PHP文件源码的编译结果缓存起来,然后在每次调用时先对比时间标记。如果未过期,则使用缓存的中间代码运行。默认缓存 3600s(一小时)。但是这样仍会浪费大量C