下面介绍了几种优化函数:
1. Extract Method (提炼函数)
解释:
如果发现一个函数的代码很长, 很可能的一种情况是这个函数做了很多事情, 找找看函数中有没有注释, 往往注释都是为了解释下面一块代码做的什么事情, 可以考虑将这块代码提炼(Extract)成一个独立的函数.
这样做的好处不言而喻, 是面向对象五大基本原则中的单一职责原则 (Single Responsibility Principle), 比较长的函数被拆分成一个个小函数, 将有利于代码被复用.
冲动前:
<ol class="dp-c"> <li class="alt"><span><span class="keyword">public</span><span> void Print(Employee employee) </span></span></li> <li><span>{ </span></li> <li class="alt"> <span class="comment">//print employee's information </span><span> </span> </li> <li> <span>Console.WriteLine(</span><span class="string">"Name:"</span><span> + employee.Name); </span> </li> <li class="alt"> <span>Console.WriteLine(</span><span class="string">"Sex:"</span><span> + employee.Sex); </span> </li> <li> <span>Console.WriteLine(</span><span class="string">"Age:"</span><span> + employee.Age); </span> </li> <li class="alt"> <span class="comment">//print employee's salary </span><span> </span> </li> <li> <span>Console.WriteLine(</span><span class="string">"Salary:"</span><span> + employee.Salary); </span> </li> <li class="alt"> <span>Console.WriteLine(</span><span class="string">"Bonus:"</span><span> + employee.Bonus); </span> </li> <li><span>} </span></li> </ol>
冲动后:
<ol class="dp-c"> <li class="alt"><span><span class="keyword">public</span><span> void Print(Employee employee) </span></span></li> <li><span>{ </span></li> <li class="alt"> <span class="comment">//print employee's information </span><span> </span> </li> <li><span>PrintInfo(employee); </span></li> <li class="alt"> <span class="comment">//print employee's salary </span><span> </span> </li> <li><span>PrintSalary(employee); </span></li> <li class="alt"><span>} </span></li> <li> <span class="keyword">public</span><span> void PrintInfo(Employee employee) </span> </li> <li class="alt"><span>{ </span></li> <li> <span>Console.WriteLine(</span><span class="string">"Name:"</span><span> + employee.Name); </span> </li> <li class="alt"> <span>Console.WriteLine(</span><span class="string">"Sex:"</span><span> + employee.Sex); </span> </li> <li> <span>Console.WriteLine(</span><span class="string">"Age:"</span><span> + employee.Age); </span> </li> <li class="alt"><span>} </span></li> <li> <span class="keyword">public</span><span> void PrintSalary(Employee employee) </span> </li> <li class="alt"><span>{ </span></li> <li> <span>Console.WriteLine(</span><span class="string">"Salary:"</span><span> + employee.Salary); </span> </li> <li class="alt"> <span>Console.WriteLine(</span><span class="string">"Bonus:"</span><span> + employee.Bonus); </span> </li> <li><span>} </span></li> </ol>
2. Inline Method (将函数内联)
解释:
有些函数很短, 只有一两行, 而且代码的意图也非常明显, 这时可以考虑将这个函数干掉, 直接使用函数中的代码.物件中过多的方法会让人感到不舒服, 干掉完全不必要的函数后代码会更简洁.
冲动前:
<ol class="dp-c"> <li class="alt"><span><span class="keyword">public</span><span> bool IsDeserving(int score) </span></span></li> <li><span>{ </span></li> <li class="alt"> <span class="keyword">return</span><span> IsScoreMoreThanSixty(score); </span> </li> <li><span>} </span></li> <li class="alt"> <span class="keyword">public</span><span> bool IsScoreMoreThanSixty(int score) </span> </li> <li><span>{ </span></li> <li class="alt"> <span class="keyword">return</span><span> (score > 60); </span> </li> <li><span>} </span></li> </ol>
冲动后:
<ol class="dp-c"> <li class="alt"><span><span class="keyword">public</span><span> bool IsDeserving(int score) </span></span></li> <li><span>{ </span></li> <li class="alt"> <span class="keyword">return</span><span> (score > 60) ; </span> </li> <li><span>} </span></li> </ol>
3. Inline Temp (将临时变量内联)
解释:
如果有一个临时变量 (Temp)用来表示某个函数的返回值, 一般来说, 这样的做法挺好的. 但如果这个临时变量实在多余, 将这个临时变量内联之后毫不影响代码的阅读, 甚至这个临时变量妨碍了其它重构工作, 就应该将这个临时变量内联化.
把这个临时变量干掉的好处在于减少了函数的长度, 有时可以让其它重构工作更顺利的进行.
冲动前:
<ol class="dp-c"> <li class="alt"><span><span>int salary = employee.Salary; </span></span></li> <li> <span class="keyword">return</span><span> (salary > 10000); </span> </li> </ol>
冲动后:
<ol class="dp-c"> <li class="alt"><span><span class="keyword">return</span><span> (employee.Salary > 10000); </span></span></li> <li><span>Replace Temp With Query (用查询式代替临时变量) </span></li> </ol>
解释:
程序中有一个临时变量(Temp)用来保存某个表达式的计算结果, 将这个计算表达式提炼(Extract)到一个独立的函数(即查询式Query)中, 将这个临时变量所有被调用的地方换成对新函数(Query)的调用, 新函数还可以被其它函数使用.
好处在于减少函数长度, 增加代码复用率, 有利于代码进一步的重构. 并且注意 Replace Temp With Query 往往是 Extract Method 之前必不可少的步骤, 因为局部变量会使代码不太容易被提炼, 所以在进行类似的重构前可以将它们替换成查询式.
下面的这个例子不是很有必要使用Replace Temp With Query, 主要展示如何 Replace Temp With Query. 试想"冲动前"函数中有很多个代码块都使用到 totalPrice, 突然有一天我发现这个函数太长, 我需要将这一块块的代码提炼成单独的函数, 这样就需要将 totalPrice = price * num; 放到每一个提炼出来的函数中. 而如果原来函数中使用的是查询式, 就不存在这个问题. 如果查询式中的计算量很大, 也不建议使用 Replace Temp With Query.
冲动前:
<ol class="dp-c"> <li class="alt"><span><span class="keyword">public</span><span> double FinalPrice(double price, int num) </span></span></li> <li><span>{ </span></li> <li class="alt"><span>double totalPrice = price * num; </span></li> <li> <span class="keyword">if</span><span> (totalPrice > 100) </span> </li> <li class="alt"> <span class="keyword">return</span><span> totalPrice * 0.8; </span> </li> <li> <span class="keyword">else</span><span> </span> </li> <li class="alt"> <span class="keyword">return</span><span> totalPrice * 0.9; </span> </li> <li><span>} </span></li> </ol>
冲动后:
<ol class="dp-c"> <li class="alt"><span><span class="keyword">public</span><span> double FinalPrice(double price, int num) </span></span></li> <li><span>{ </span></li> <li class="alt"> <span class="keyword">if</span><span> (TotalPrice(price, num) > 100) </span> </li> <li> <span class="keyword">return</span><span> TotalPrice(price, num) * 0.8; </span> </li> <li class="alt"> <span class="keyword">else</span><span> </span> </li> <li> <span class="keyword">return</span><span> TotalPrice(price, num) * 0.9; </span> </li> <li class="alt"><span>} </span></li> <li> <span class="keyword">public</span><span> double TotalPrice(double price, int num) </span> </li> <li class="alt"><span>{ </span></li> <li> <span class="keyword">return</span><span> price * num; </span> </li> <li class="alt"><span>} </span></li> </ol>
5. Introduce Explaining Variable (引入可以理解的变量)
解释:
很多时候在条件逻辑表达式中, 很多条件令人难以理解它的意义, 为什么要满足这个条件? 不清楚. 可以使用Introduce Explaining Variable将每个条件子句提炼出来, 分别用一个恰当的临时变量名表示条件子句的意义.
好处在于增加了程序的可读性.
冲动前:
<ol class="dp-c"> <li class="alt"><span><span class="keyword">if</span><span>((operateSystem.Contains(</span><span class="string">"Windows"</span><span>))&& (browser.Contatins(</span><span class="string">"IE"</span><span>))) </span></span></li> <li><span>{ </span></li> <li class="alt"> <span> </span><span class="comment">//do something </span><span> </span> </li> <li><span>} </span></li> </ol>
冲动后:
<ol class="dp-c"> <li class="alt"><span><span>bool isWindowsOS = operateSystem.Contains(</span><span class="string">"Windows"</span><span>); </span></span></li> <li> <span>bool isIEBrowser = browser.Contatins(</span><span class="string">"IE"</span><span>); </span> </li> <li class="alt"> <span class="keyword">if</span><span> (isWindowsOS && isIEBrowser) </span> </li> <li><span>{ </span></li> <li class="alt"> <span class="comment">//do something </span><span> </span> </li> <li><span>} </span></li> </ol>
6. Split Temporary Variable (撇清临时变量)
解释:
例如代码中有个临时变量在函数上面某处表示长方形周长, 在函数下面被赋予面积, 也就是这个临时变量被赋值超过一次, 且表示的不是同一种量. 应该针对每次赋值, 分配一个独立的临时变量.
一个变量只应表示一种量, 否则会令代码阅读者感到迷惑.
冲动前:
<ol class="dp-c"> <li class="alt"><span><span>double temp = (width + height) * 2; </span></span></li> <li> <span class="comment">//do something </span><span> </span> </li> <li class="alt"><span>temp = width * height; </span></li> <li> <span class="comment">//do something </span><span> </span> </li> </ol>
冲动后:
<ol class="dp-c"> <li class="alt"><span><span>double perimeter = (width + height) * 2; </span></span></li> <li> <span class="comment">//do something </span><span> </span> </li> <li class="alt"><span>double area = width * height; </span></li> <li> <span class="comment">//do something </span><span> </span> </li> </ol>
7. Remove Assignments to Parameters (消除对参数的赋值操作)
解释:
传入参数分"传值"和"传址"两种, 如果是"传址", 在函数中改变参数的值无可厚非, 因为我们就是想改变原来的值. 但如果是"传值", 在代码中为参数赋值, 就会令人产生疑惑. 所以在函数中应该用一个临时变量代替这个参数, 然后对这个临时变量进行其它赋值操作.
冲动前:
<ol class="dp-c"> <li class="alt"><span><span class="keyword">public</span><span> double FinalPrice(double price, int num) </span></span></li> <li><span>{ </span></li> <li class="alt"><span>price = price * num; </span></li> <li> <span class="comment">//other calculation with price </span><span> </span> </li> <li class="alt"> <span class="keyword">return</span><span> price; </span> </li> <li><span>} </span></li> </ol>
冲动后:
<ol class="dp-c"> <li class="alt"><span><span class="keyword">public</span><span> double FinalPrice(double price, int num) </span></span></li> <li><span>{ </span></li> <li class="alt"><span>double finalPrice = price * num; </span></li> <li> <span class="comment">//other calculation with finalPrice </span><span> </span> </li> <li class="alt"> <span class="keyword">return</span><span> finalPrice; </span> </li> <li><span>} </span></li> </ol>
8. Replace Method with Method Object (用函数物件代替函数)
解释:
冲动的写下一行行代码后, 突然发现这个函数变得非常大, 而且由于这个函数包含了很多局部变量, 使得无法使用 Extract Method, 这时 Replace Method with Method Object 就起到了杀手锏的效果. 做法是将这个函数放入一个单独的物件中, 函数中的临时变量就变成了这个物件里的值域 (field).
冲动前:
<ol class="dp-c"> <li class="alt"><span><span class="keyword">class</span><span> Bill </span></span></li> <li><span>{ </span></li> <li class="alt"> <span class="keyword">public</span><span> double FinalPrice() </span> </li> <li><span>{ </span></li> <li class="alt"><span>double primaryPrice; </span></li> <li><span>double secondaryPrice; </span></li> <li class="alt"><span>double teriaryPrice; </span></li> <li> <span class="comment">//long computation </span><span> </span> </li> <li class="alt"><span>... </span></li> <li><span>} </span></li> <li class="alt"><span>} </span></li> </ol>
冲动后:
<ol class="dp-c"> <li class="alt"><span><span class="keyword">class</span><span> Bill </span></span></li> <li><span>{ </span></li> <li class="alt"> <span class="keyword">public</span><span> double FinalPrice() </span> </li> <li><span>{ </span></li> <li class="alt"> <span class="keyword">return</span><span> </span><span class="keyword">new</span><span> PriceCalculator(this).compute(); </span> </li> <li><span>} </span></li> <li class="alt"><span>} </span></li> <li> <span class="keyword">class</span><span> PriceCalculator </span> </li> <li class="alt"><span>{ </span></li> <li><span>double primaryPrice; </span></li> <li class="alt"><span>double secondaryPrice; </span></li> <li><span>double teriaryPrice; </span></li> <li class="alt"> <span class="keyword">public</span><span> PriceCalculator(Bill bill) </span> </li> <li><span>{ </span></li> <li class="alt"> <span class="comment">//initial </span><span> </span> </li> <li><span>} </span></li> <li class="alt"> <span class="keyword">public</span><span> double compute() </span> </li> <li><span>{ </span></li> <li class="alt"> <span class="comment">//computation </span><span> </span> </li> <li><span>} </span></li> <li class="alt"><span>} </span></li> </ol>
9. Substitute Algorithm (替换算法)
解释:
有这么一个笑话:
某跨国日化公司, 肥皂生产线存在包装时可能漏包肥皂的问题, 肯定不能把空的肥皂盒卖给顾客, 于是该公司总裁命令组成了以博士牵头的专家组对这个问题进行攻关, 该研发团队使用了世界上最高精尖的技术 (如红外探测, 激光照射等), 在花费了大量美金和半年的时间后终于完成了肥皂盒检测系统, 探测到空的肥皂盒以后, 机械手会将空盒推出去. 这一办法将肥皂盒空填率有效降低至5%以内, 问题基本解决.
而某乡镇肥皂企业也遇到类似问题, 老板命令初中毕业的流水线工头想办法解决之, 经过半天的思考, 该工头拿了一台电扇到生产线的末端对着传送带猛吹, 那些没有装填肥皂的肥皂盒由于重量轻就都被风吹下去了...
这个笑话可以很好的解释 Substitute Algorithm, 对于函数中复杂的算法, 尽量想办法将这个算法简单化, 从而达到与之前同样甚至更好的效果.
本文链接:

在PHP中,trait適用於需要方法復用但不適合使用繼承的情況。 1)trait允許在類中復用方法,避免多重繼承複雜性。 2)使用trait時需注意方法衝突,可通過insteadof和as關鍵字解決。 3)應避免過度使用trait,保持其單一職責,以優化性能和提高代碼可維護性。

依賴注入容器(DIC)是一種管理和提供對象依賴關係的工具,用於PHP項目中。 DIC的主要好處包括:1.解耦,使組件獨立,代碼易維護和測試;2.靈活性,易替換或修改依賴關係;3.可測試性,方便注入mock對象進行單元測試。

SplFixedArray在PHP中是一種固定大小的數組,適用於需要高性能和低內存使用量的場景。 1)它在創建時需指定大小,避免動態調整帶來的開銷。 2)基於C語言數組,直接操作內存,訪問速度快。 3)適合大規模數據處理和內存敏感環境,但需謹慎使用,因其大小固定。

PHP通過$\_FILES變量處理文件上傳,確保安全性的方法包括:1.檢查上傳錯誤,2.驗證文件類型和大小,3.防止文件覆蓋,4.移動文件到永久存儲位置。

JavaScript中處理空值可以使用NullCoalescingOperator(??)和NullCoalescingAssignmentOperator(??=)。 1.??返回第一個非null或非undefined的操作數。 2.??=將變量賦值為右操作數的值,但前提是該變量為null或undefined。這些操作符簡化了代碼邏輯,提高了可讀性和性能。

CSP重要因為它能防範XSS攻擊和限制資源加載,提升網站安全性。 1.CSP是HTTP響應頭的一部分,通過嚴格策略限制惡意行為。 2.基本用法是只允許從同源加載資源。 3.高級用法可設置更細粒度的策略,如允許特定域名加載腳本和样式。 4.使用Content-Security-Policy-Report-Only頭部可調試和優化CSP策略。

HTTP請求方法包括GET、POST、PUT和DELETE,分別用於獲取、提交、更新和刪除資源。 1.GET方法用於獲取資源,適用於讀取操作。 2.POST方法用於提交數據,常用於創建新資源。 3.PUT方法用於更新資源,適用於完整更新。 4.DELETE方法用於刪除資源,適用於刪除操作。

HTTPS是一種在HTTP基礎上增加安全層的協議,主要通過加密數據保護用戶隱私和數據安全。其工作原理包括TLS握手、證書驗證和加密通信。實現HTTPS時需注意證書管理、性能影響和混合內容問題。


熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

Safe Exam Browser
Safe Exam Browser是一個安全的瀏覽器環境,安全地進行線上考試。該軟體將任何電腦變成一個安全的工作站。它控制對任何實用工具的訪問,並防止學生使用未經授權的資源。

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

Atom編輯器mac版下載
最受歡迎的的開源編輯器

SublimeText3 英文版
推薦:為Win版本,支援程式碼提示!

記事本++7.3.1
好用且免費的程式碼編輯器