一起谈.NET技术,改善代码设计 —— 简化条件表达式(Simplifying Conditional Expressions)

简介:   系列博客      1. 改善代码设计 —— 优化函数的构成(Composing Methods)      2. 改善代码设计 —— 优化物件之间的特性(Moving Features Between Objects)      3.

  系列博客

      1. 改善代码设计 —— 优化函数的构成(Composing Methods)

      2. 改善代码设计 —— 优化物件之间的特性(Moving Features Between Objects)

      3. 改善代码设计 —— 组织好你的数据(Composing Data)

      4. 改善代码设计 —— 简化条件表达式(Simplifying Conditional Expressions)

      5. 改善代码设计 —— 简化函数调用(Making Method Calls Simpler)

      6. 改善代码设计 —— 处理概括关系(Dealing with Generalization)

  1. Decompose Conditional (分解条件式)

  解释:

      "复杂的条件逻辑" 是导致复杂性上升最常见的地方, "条件表达式中堆积的计算过程", "条件式表达得不简洁"等等都是造成复杂的原因. Decompose Conditional 用于将这些复杂的元素从条件表达式中分离出去, 仅在条件表达式中调用简洁的函数.

      这样做带来的直接好处是减少重复, 而且代码的可读性提高了.

  冲动前:

 
 
if (date.After(SUMMER_START) && date.Before(SUMMER_END))
charge
= days * _price + _summerServiceTip;
else
charge
= days * _price;

  冲动后:

 
 
if (date.IsSummer())
charge
= SummerCharge(days);
else
charge
= WinterCharge(days);

  2. Consolidate Conditional Expression (合并条件式)

  解释:

      如果代码中有一连串的 if 检查语句, 检查语句中的条件不相同, 但最终的行为都是一样的. 对于这样的情况, 应该使用 "逻辑与" 和 "逻辑或" 将它们合并成一个条件表达式, 如果嫌这个合并条件后的表达式太罗嗦, 你还可以将这个表达式提取成一个函数.

  冲动前:

 
 
if (computer.CPU != "T6670")
return false;
if (computer.RAM != " 1 .00GB")
return false;
if (computer.SytemType != " 32 - bit Operating System")
return false;
// other compution

  冲动后:

 
 
if ((computer.CPU != "T6670") || (computer.RAM != " 1 .00GB") || (computer.SytemType != " 32 - bit Operating System"))
return false;
// other compution

      你还可以将 if 里长长的条件表达式提取成一个方法, 如 bool IsStandard(Computer computer), 这样在原来的 if 语句中只需要调用这个方法即可微笑

  3. Consolidate Duplicate Conditional Fragments (合并重复的条件片段)

  解释:

      如果条件式的每个分支上都有同样一段代码, 如果这段代码对条件分支在执行这段代码后执行后面的代码没有影响,  请将这段代码移到条件式的外面.

  冲动前:

 
 
if (date.IsSummer())
{
charge
= days * _price + _summerServiceTip;
PrintDetail();
}
else
{
charge
= days * _price;
PrintDetail();
}
// other compution

  冲动后:

 
 
charge = days * _price;

if (date.IsSummer())
charge
+= _summerServiceTip;

PrintDetail();
// other compution

  4. Remove Control Flag (移除控制标志)

  解释:

      很多代码里执行一个 for 或者 while 循环用于寻找一个数组里特点的元素, 很多时候在循环开头就执行控制标志的检查, 满足检查条件就继续执行循环查找元素. 如果这一次查找到了想要的元素, 就更改控制标志的值, 让它下次被检查出不符合条件, 从而循环结束.

      这并不是一个很好的做法, 使用诸如 break, continue, return 语句会让你的代码意图更加直接, 更加明显.

  冲动前:

 
 
for ( int i = 0 ; i < suspects.Length; i ++ )
{
if (!found)
{
if (suspects [ i ] .Name == guessName)
{
sendAlert();
found
= true;
}
}
}

  冲动后:

 
 
for ( int i = 0 ; i < suspects.Length; i ++ )
{
if (suspects [ i ] .Name == guessName)
{
sendAlert();
break ;
}
}

  5. Replace Nested Conditional with Guard Clauses (以卫语句取代嵌套条件式)

  解释:

      许多程序员觉得函数应该只有一个出口 (return), 结果导致函数中的条件逻辑 (Conditional Logic) 本来完全可以终止下面的代码继续执行 (因为没有必要), 结果却只在函数最后 return, 使人难以看清程序的执行路径.

      Replace Nested Conditional with Guard Clauses 用来解决这个问题, 它能带给代码可读性的提高, 还有性能上一点点的优化.

  冲动前:

 
 
double charge;

if (IsSummer(date))
{
// ...
SummerCharge(charge);
}
else
{
// ...
WinterCharge(charge);
}

return charge;

  冲动后:

 
 
double charge;

if (IsSummer(date))
{
// ...
SummerCharge(charge);
return charge;
}
else
{
// ...
WinterCharge(charge);
return charge;
}

  6. Replace Conditional with Polymorphism (以多态取代条件)

  解释:

      这条重构手法常常用于消除函数中长长的 switch-case 语句. 虽然写一个个的子类比较繁琐, 但随着项目的进行, 好处会体现出来的.

  冲动前:

 
 
public double Salary(Employee employee)
{
switch(employee.Type):
{
case Employee.Engineer
{
// ...
}
case Employee.Salesman:
{
// ...
}
// ...
default :
{
// ...
}
}
}

  冲动后:

 
 
public abstract double Salary(Employee employee);

class Engineer : Employee
{
public override double Salary(Employee employee)
{
// ...
}
}
class Salesman : Employee
{
public override double Salary(Employee employee)
{
// ...
}
}

  7. Introduce Null Object (引入 Null 对象)

  解释:

      如果代码中出现很多判断某值是不是为 null , 诸如 if (XXX != null) {//...} else {//...} 这样的情况, 可以考虑使用 Introduce Null Object 重构手段. 这个手段其实并不难以理解, 可以简单理解成为某一个物件在为空状态下设定默认的值域和行为, 可以建立一个子类, 继承父类中需要对 "为空" 情况下做出响应的虚函数或者值域. 它是 Null Object 设计模式里的最基础最常见的手段.

  8. Introduce Assertion (引入断言)

  解释:

      严格上说, 引入断言并不是为了简化条件表达式, 它主要是为了代替条件表达式上面的注释, 通常这样的注释用来解释下面的条件表达式是基于什么样的假设之上的. 通常经过一系列的测试, 发现所写的断言在任何情况下都是正确的, 在系统发布的时候可以把它们全部删除掉.

      在 C# 中引入断言使用 Debug.Assert() 方法, 如果一切假设都是正确的, 则代码会顺利的进行.

  冲动前:

 
 
// index should between 0 to 10
return (customers [ index ] == "James") ? true : false;

  冲动后:

 
 
Debug.Assert(( index >= 0 ) && ( index <= 10 ), "Error", " index should between 0 to 10 ");
return (customers [ index ] == "James") ? true : false;

  如果断言错误, 在运行的时候会有一个消息框给予错误信息的提示.

目录
相关文章
|
25天前
|
人工智能 开发框架 C#
C#/.NET/.NET Core技术前沿周刊 | 第 6 期(2024年9.16-9.22)
C#/.NET/.NET Core技术前沿周刊 | 第 6 期(2024年9.16-9.22)
|
25天前
|
人工智能 开发框架 Cloud Native
C#/.NET/.NET Core技术前沿周刊 | 第 9 期(2024年10.07-10.13)
C#/.NET/.NET Core技术前沿周刊 | 第 9 期(2024年10.07-10.13)
|
25天前
|
数据可视化 NoSQL C#
C#/.NET/.NET Core技术前沿周刊 | 第 8 期(2024年10.01-10.06)
C#/.NET/.NET Core技术前沿周刊 | 第 8 期(2024年10.01-10.06)
|
25天前
|
设计模式 开发框架 C#
C#/.NET/.NET Core技术前沿周刊 | 第 4 期(2024年9.1-9.8)
C#/.NET/.NET Core技术前沿周刊 | 第 4 期(2024年9.1-9.8)
|
2月前
|
人工智能 前端开发 开发工具
解读.NET 技术的开发潜力
本文全面介绍了.NET技术在软件开发领域的核心优势、创新应用及面临的挑战。.NET以其统一的开发平台、强大的工具和跨平台能力,成为企业级应用、Web应用乃至游戏开发的理想选择。然而,在性能优化、容器化及AI集成等方面仍需不断突破。通过积极拥抱开源和社区驱动模式,.NET将持续推动软件开发的进步。
54 1
|
2月前
|
人工智能 前端开发 Devops
.NET技术自发布以来,在软件开发领域发挥了重要作用
【9月更文挑战第12天】.NET技术自发布以来,在软件开发领域发挥了重要作用。本文分为三部分探讨其在现代开发中的应用:首先介绍.NET的核心价值,包括语言多样性、强大的开发工具支持、丰富的类库、跨平台能力和活跃的社区;接着分析其在企业级应用、Web开发、移动应用、云服务及游戏开发中的实际应用;最后讨论.NET面临的挑战与未来趋势,如性能优化、容器化、AI集成及跨平台框架竞争等。通过不断的技术创新和社区驱动,.NET将持续推动软件开发的进步。
38 4
|
2月前
|
传感器 应用服务中间件 Linux
C#/.NET/.NET Core技术前沿周刊 | 第 3 期(2024年8.26-8.31)
C#/.NET/.NET Core技术前沿周刊 | 第 3 期(2024年8.26-8.31)
|
2月前
|
开发框架 前端开发 JavaScript
ASP.NET MVC 教程
ASP.NET 是一个使用 HTML、CSS、JavaScript 和服务器脚本创建网页和网站的开发框架。
38 7
|
2月前
|
存储 开发框架 前端开发
ASP.NET MVC 迅速集成 SignalR
ASP.NET MVC 迅速集成 SignalR
53 0
|
3月前
|
开发框架 前端开发 .NET
ASP.NET MVC WebApi 接口返回 JOSN 日期格式化 date format
ASP.NET MVC WebApi 接口返回 JOSN 日期格式化 date format
46 0