<style>
.textleft {
text-align:left;
}
.reveal, .reveal h1, .reveal h2, .reveal h3, .reveal h4, .reveal h5, .reveal h6 {
font-family:Arial, Microsoft JhengHei;}
.small-font {
font-size: 20px !important;
}
.reveal .progress {
height: 14px !important;
}
.progress span {
background: url() repeat-x !important;
}
.progress span:after, .progress span.nyancat {
content: "";
background: url() ;
width: 36px !important;
height: 21px !important;
border: none !important;
float: right;
margin-top: -7px;
margin-right: -10px;
transform: scale(1.5,1.5);
}
.reveal section img {
height: 550px !important;
}
</style>
<!-- .slide: data-transition="slide" -->
# Clean Code 讀書筆記 - Chapter 2 有意義的命名
@Kais(VagrantPi)
###### tags: `slide`, `簡報`, `Clean Code`
---
<!-- .slide: data-transition="slide" -->
## Agenda
- 讓名稱代表意圖 -- 使之名副其實
- 避免誤導
- 產生有意義的區別
- 能被唸出來的名稱
- 使用可被搜尋的名字
- 避免編碼
- 避免思維轉換
- 類別的命名
- 方法的命名
----
<!-- .slide: data-transition="convex" -->
- 不要裝可愛
- 每個概念使用一個字詞
- 別說雙關語
- 使用解決方案領域命名
- 使用問題領域的命名
- 添加有意義的上下文資訊(Context)
- 別添加無理由的上下文資訊
- 總結
- Q&A
- 該章節可否優化團隊現狀
---
<!-- .slide: data-transition="convex" -->
## 讓名稱代表意圖 -- 使之名副其實
----
<!-- .slide: data-transition="convex" -->
```java
int d; // elapsed time in days
```
```java
int elapsedTimeInDays;
int daysSinceCreation;
int daysSinceModification;
int fileAgeInDays;
```
----
<!-- .slide: data-transition="convex" -->
```java
public List<int[]> getThem() {
List<int[]> list1 = new ArrayList<int[]>();
// theList 代表什麼?
for (int[] x : theList)
// x[0] 代表什麼?
// 4 代表什麼?
if (x[0] == 4)
list1.add(x);
return list1;
}
```
----
<!-- .slide: data-transition="convex" -->
看得出來這邊會遍歷 gameBoard,如果 cell 的狀態碼為 FLAGGED,則會加到已標記(flaggedCells)list 中
```diff
- public List<int[]> getThem() {
+ public List<int[]> getFlaggedCells() {
- List<int[]> list1 = new ArrayList<int[]>();
+ List<int[]> flaggedCells = new ArrayList<int[]>();
- for (int[] x : theList)
+ for (int[] cell : gameBoard)
- if (x[0] == 4)
+ if (cell[STATUS_VALUE] == FLAGGED)
- list1.add(x);
+ flaggedCells.add(cell);
- return list1;
+ return flaggedCells;
}
```
----
這邊使用泛型進一步抽象化
<!-- .slide: data-transition="convex" -->
```diff
- public List<int[]> getFlaggedCells() {
+ public List<Cell> getFlaggedCells() {
- List<int[]> flaggedCells = new ArrayList<int[]>();
+ List<Cell> flaggedCells = new ArrayList<Cell>();
for (Cell cell : gameBoard)
- if (cell[STATUS_VALUE] == FLAGGED)
+ if (cell.isFlagged())
flaggedCells.add(cell);
return flaggedCells;
}
```
---
<!-- .slide: data-transition="convex" -->
## 避免誤導
----
<!-- .slide: data-transition="convex" -->
- 應避免專有系統中的專有名詞『縮寫』(hp、aix、sco)
另外這些命名也與一些系統平台或指令撞名
----
<!-- .slide: data-transition="convex" -->
- 清單類的變數,竟量不要出現 List,可能造成混淆
- 命名過長情況下,避免出現些微差別的其他命名
- XYZControllerForEfficient`Handling`OfStrings
- XYZControllerForEfficient`Storage`OfStrings
- 避免拼寫不一致,或是容易誤導的字
- O、0、l、1,單獨或偶而出現可能還好,不過很容易混淆
---
<!-- .slide: data-transition="convex" -->
## 產生有意義的區別
----
<!-- .slide: data-transition="convex" -->
- 在命名時使用數字序列或是無意義命名時,無法提供太多作者意圖
```diff
- public static void copyChars(char a1[], char a2[]) {
+ public static void copyChars(source a1[], destination a2[]) {
for (int i = 0; i < a1.length; i++) {
a2[i] = a1[i];
}
}
```
----
<!-- .slide: data-transition="convex" -->
- 下面命名幾乎是無意義的,如同英文中的冠詞 `a`、`an`、`the`
- `Data` 與 `Info` 之類的用詞也是無意義的,舉例 `ProductData`、`ProductInfo` 同樣代表 Product 類別的資料,但兩者差異只在於命名不同,幾乎等價於 `Product`
- variable 字眼出現在變數命名
- table 字眼出現在表格名稱
- Customer 與 CustomerObject 差別在哪,哪個最能代表客戶付款紀錄
---
<!-- .slide: data-transition="convex" -->
## 能被唸出來的名稱
----
<!-- .slide: data-transition="convex" -->
常見縮寫併再一起幾乎唸不出來
genymdhms (generation date, year, month, day, hour, minute,
and second)
都念不出來了,更難想像意圖,因此沒有文件下,後續接手的同人無法維護
----
<!-- .slide: data-transition="convex" -->
應該可以簡單判斷哪個比較好讀
```java
class DtaRcrd102 {
private Date genymdhms;
private Date modymdhms;
private final String pszqint = "102";
/* ... */
};
```
```java
class Customer {
private Date generationTimestamp;
private Date modificationTimestamp;;
private final String recordId = "102";
/* ... */
};
```
---
<!-- .slide: data-transition="convex" -->
## 使用可被搜尋的名字
----
<!-- .slide: data-transition="convex" -->
- 長命名勝於短命名
- 常數命名會避數字容易被搜尋
```java
const int WORK_DAYS_PER_WEEK = 5;
```
比較要在整個專案中搜尋到 5,`WORK_DAYS_PER_WEEK` 會更好找到
---
<!-- .slide: data-transition="convex" -->
## 避免編碼
----
<!-- .slide: data-transition="convex" -->
- [匈牙利命名法
](https://zh.wikipedia.org/zh-tw/%E5%8C%88%E7%89%99%E5%88%A9%E5%91%BD%E5%90%8D%E6%B3%95)
- 會在字首加上型別,但其實在強型別語言中,就是無意義的規定
- 成員字首(Member Prefixes)
- 不需要再變數前面新增 m_aaa, m_bbb 的前綴
----
<!-- .slide: data-transition="convex" -->
- Interface、Implementations
介面的命名,比起使用 `IShapeFactory` 更傾向使用 `ShapeFactory`,而如果一定要對 Interface、Implementations 進行命名,這樣這邊會建議只對 Implementations 命名。`ShapeFactoryImp` 都比在 interface 在前面加個 I 好
---
<!-- .slide: data-transition="convex" -->
## 避免思維轉換
----
<!-- .slide: data-transition="convex" -->
`i`、`j`、`k` 這類變數,使用在 for 迴圈內可說是約定熟成
但單個字當變數在其他地方,可以說是非常糟糕,還需要在腦內轉換一下,才能對應到真實的變數概念
『清楚才是王道』
---
<!-- .slide: data-transition="convex" -->
## 類別的命名
----
<!-- .slide: data-transition="convex" -->
class 的命名應該為名詞與名詞片語
- ⭕️ Customer(顧客)
- ⭕️ WikiPage(維基頁面)
- ⭕️ Account(帳戶)
- ⭕️ AddressParser(地址解析器)
- ❌ Manager(管理者)
- ❌ Processor(處理者)
- ❌ Data(資料)
- ❌ Info(資訊)
---
<!-- .slide: data-transition="convex" -->
## 方法的命名
----
<!-- .slide: data-transition="convex" -->
方法應該是用動詞或是動詞片語
- postPayment(登記款項)
- deletePage(刪除頁面)
- save(儲存)
----
<!-- .slide: data-transition="convex" -->
此外,在 Javabean 的標準中
- accessors(取出器)使用 get 當字首
- mutators(修改器)使用 set 當字首
- predicates(判定)使用 is 當字首
---
<!-- .slide: data-transition="convex" -->
## 不要裝可愛
----
<!-- .slide: data-transition="convex" -->
HolyHandGrenade(神聖手榴彈),光看名字不知道這個 Method 要做什麼,娛樂價值大於實用
---
<!-- .slide: data-transition="convex" -->
## 每個概念使用一個字詞
----
<!-- .slide: data-transition="convex" -->
- fetch、retrieve、get,都可以用在取得的方法命名,但是如果沒有統一,在使用時會需要而外花時間理解
- controller(控制器)、manager(管理員)、driver(驅動程式)也不要一起使用,
---
<!-- .slide: data-transition="convex" -->
## 別說雙關語
----
<!-- .slide: data-transition="convex" -->
原先有一個 add 方法用來做『相加或想連』的概念,散落多個 class 中,今天有一個新類別也用了這個 add,但是功能卻將參數『加』進物件內。因此這出現了一個命名兩個意思的問題,這邊可能使用 insert 或是 append 會更好些
---
<!-- .slide: data-transition="convex" -->
## 使用解決方案領域命名
----
<!-- .slide: data-transition="convex" -->
使用工程師的電腦科學專業術語、演算法名稱、pattern name、數學字彙協助命名
因為讀你程式碼的,也會是工程師
---
<!-- .slide: data-transition="convex" -->
## 使用問題領域的命名
----
<!-- .slide: data-transition="convex" -->
可以詢問問題領域專家使用的術語來幫助命名
---
<!-- .slide: data-transition="convex" -->
## 添加有意義的上下文資訊(Context)
----
<!-- .slide: data-transition="convex" -->
變數時常需要搭配上下文,來盼端該變數實際的意義
如果只看到變數 state ,你會聯想到什麼
----
<!-- .slide: data-transition="convex" -->
那看到 street, houseNumber, city, state, zipcode 你應該猜得出來這個 state 是什麼了
當然,如果將這些全部幫進 Address 的 class 會更清楚
----
<!-- .slide: data-transition="convex" -->
以下有另外一個例子
```java
// 統計猜測
private void printGuessStatistics(char candidate, int count) {
// 數字
String number;
// 動詞
String verb;
// 複數型態修改器
String pluralModifier;
if (count == 0) {
number = "no";
verb = "are";
pluralModifier = "s";
} else if (count == 1) {
number = "1";
verb = "is";
pluralModifier = "";
} else {
number = Integer.toString(count);
verb = "are";
pluralModifier = "s";
}
String guessMessage = String.format(
"There %s %s %s%s", verb, number, candidate, pluralModifier
);
print(guessMessage);
}
```
----
<!-- .slide: data-transition="convex" -->
這邊提到了幾個問題點
- 函數過長
- 所有區域變數在整段函數中都被使用
```java
public class GuessStatisticsMessage {
private String number;
private String verb;
private String pluralModifier;
public String make(char candidate, int count) {
createPluralDependentMessageParts(count);
return String.format(
"There %s %s %s%s",
verb, number, candidate, pluralModifier );
}
private void createPluralDependentMessageParts(int count) {
if (count == 0) {
thereAreNoLetters();
} else if (count == 1) {
thereIsOneLetter();
} else {
thereAreManyLetters(count);
}
}
private void thereAreManyLetters(int count) {
number = Integer.toString(count);
verb = "are";
pluralModifier = "s";
}
private void thereIsOneLetter() {
number = "1";
verb = "is";
pluralModifier = "";
}
private void thereAreNoLetters() {
number = "no";
verb = "are";
pluralModifier = "s";
}
}
```
---
<!-- .slide: data-transition="convex" -->
## 別添加無理由的上下文資訊
----
<!-- .slide: data-transition="convex" -->
假設有個系統『豪華版加油站(Gas Station Deluxe)』
因此在每個 Class 前都加上 GSD 字首,這很明顯在跟 IDE 做對,因為當輸入 `G` 後,會跳出系統哪所影類別的建議
----
<!-- .slide: data-transition="convex" -->
此外,如果較小的命名可以清楚表達,則好過較長命名,以免添加過多與內文不符的名稱
```java
public class Address {
private String accountAddress; // 帳戶地址
private String customerAddress; // 客戶地址
}
```
`accountAddress`、`customerAddress`是個明確的命名
---
<!-- .slide: data-transition="convex" -->
## 總結
----
<!-- .slide: data-transition="convex" -->
需要良好的描述技巧,與共同的文化背景,
---
<!-- .slide: data-transition="convex" -->
## Q&A
---
<!-- .slide: data-transition="convex" -->
## 該章節可否優化團隊現狀
---
{"metaMigratedAt":"2023-06-18T02:53:12.086Z","metaMigratedFrom":"Content","title":"Clean Code 讀書筆記 - Chapter 2 有意義的命名","breaks":true,"contributors":"[{\"id\":\"69ade472-3ed3-499d-8a69-767243a31621\",\"add\":9106,\"del\":320}]","description":"@Kais(VagrantPi)"}