# 基本資料型別® 導讀人:Jim ---- <div style="display: flex; justify-content: center;align-items: center"> <img src="https://hackmd.io/_uploads/rkN8zs-QA.jpg" width="50%" style="margin-right: 50px;" /> <div width="50%"> <h4>基本資料型態®</h4> <ul style="text-align: left; margin-left: -120px; font-size:24px"> <li>數值概論</li> <li>整數</li> <li>浮點數</li> <li>字元和字串</li> <li>布林變數</li> <li>列舉型別</li> <li>具名常數</li> <li>陣列</li> <li>建立型別</li> <li>...</li> </ul> <img src="https://hackmd.io/_uploads/HkCAKsWQC.png" width="200px" style="position: absolute;right:20px;bottom: 10px" /> </div> <div> ---- <div style="display: flex; justify-content: center;align-items: center"> <img src="https://hackmd.io/_uploads/rkN8zs-QA.jpg" width="50%" style="margin-right: 50px;" /> <div width="50%"> <h4>基本資料型態®</h4> <ul style="text-align: left; margin-left: -120px; font-size:24px; list-style: none"> <li style="color: #FCD354; list-style-image: url('https://hackmd.io/_uploads/Bk5lGONQC.png');background-size: 5px;"><b>數值概論</b></li> <li>整數</li> <li>浮點數</li> <li>字元和字串</li> <li>布林變數</li> <li>列舉型別</li> <li>具名常數</li> <li>陣列</li> <li>建立型別</li> <li>...</li> </ul> <img src="https://hackmd.io/_uploads/HkCAKsWQC.png" width="200px" style="position: absolute;right:20px;bottom: 10px" /> </div> <div> ---- ## <h3 style="text-align:left;"><img src="https://hackmd.io/_uploads/HJMQOONXC.png" width="45px" style="margin-top: 3px;margin-right: 10px;"><span style="color: #FCD354;">數值概論</span></h3> <ul> <li>避免使用魔術數值(magic number)</li> <li>如果需要,可以使用硬編碼的 0 和 1 (hard-coding 的數字只允許 0 或 1)</li> <li>預防除以零(Division-by-zero)錯誤</li> <li>使型別轉換變得明顯</li> <li>避免混和型別的比較</li> <li>注意編譯器的警告</li> </ul> ---- #### 避免使用魔術數值(magic number) ```javascript= // 修改前 def calculate_final_price(price): # 0.08 是一個 magic number,代表 8% 的銷售稅率 tax = price * 0.08 final_price = price + tax return final_price # 這裡的 100 是商品的原價 final_price = calculate_final_price(100) print("The final price with tax is:", final_price) ``` ---- #### 避免使用魔術數值(magic number) ```javascript= // 修改後 TAX_RATE = 0.08 # 8% 的銷售稅率,清晰明確的命名 def calculate_final_price(price): tax = price * TAX_RATE final_price = price + tax return final_price final_price = calculate_final_price(100) print("The final price with tax is:", final_price) ``` ---- #### 如果需要,可以使用硬編碼的 0 和 1 ```= // 計算從 1 到 10 的總和 initTotal = 0 countStart = 1 countEnd = 10 def forward_addition(n): total = initTotal for i in range(countStart, n+1): total += i # 將 i 的值加到 total 上 print(f"Adding {i}, total now is {total}") return total sum_of_numbers = forward_addition(countEnd) print(f'The sum of numbers from {countStart} to {countEnd} is:', sum_of_numbers) ``` ---- #### 如果需要,可以使用硬編碼的 0 和 1 ```diff= // 計算從 1 到 10 的總和 - initTotal = 0 - countStart = 1 - countEnd = 10 def forward_addition(n): - total = initTotal + total = 0 - for i in range(countStart, n+1): + for i in range(1, n+1): total += i # 將 i 的值加到 total 上 print(f"Adding {i}, total now is {total}") return total - sum_of_numbers = forward_addition(countEnd) + sum_of_numbers = forward_addition(10) - print(f'The sum of numbers from {countStart} to {countEnd} is:', sum_of_numbers) + print(f'The sum of numbers from 0 to 10 is:', sum_of_numbers) ``` ---- #### 如果需要,可以使用硬編碼的 0 和 1 ```= // 計算從 1 到 10 的總和 def forward_addition(n): total = 0 for i in range(1, n+1): total += i # 將 i 的值加到 total 上 print(f"Adding {i}, total now is {total}") return total sum_of_numbers = forward_addition(10) print(f'The sum of numbers from 0 to 10 is:', sum_of_numbers) ``` ---- #### 預防除以零(Division-by-zero)錯誤 ![KRpI3](https://hackmd.io/_uploads/Bkr3iY47C.jpg) <p style="text-align: left">例:Pizza 分成兩份,等於 1 除以 2 等於 1/2</p> <p style="text-align: left">問:Pizza 如何分成 0 份? 1/0 = ?</p> ---- #### 使型別轉換變得明顯 ``` let num = "5"; let result = num + 1; console.log(result); // 輸出 "51" ``` ---- #### 使型別轉換變得明顯 ```= let num = "5"; let result = +num + 1; console.log(result); // 輸出 6 ``` ---- #### 使型別轉換變得明顯 ```diff= let num = "5"; - let result = +num + 1; + let result = Number(num) + 1; console.log(result); // 輸出 6 ``` ---- #### 避免混和型別的比較 ```javascript= const x = 1 const s = "1" if(x == s) { console.log("Two values are equal") ✅ } else { console.log("Two values are not equal") } // example2: console.log("0"==false) //true ``` ---- #### 避免混和型別的比較 ```javascript= const x = 1 const s = "1" if(x === s) { console.log("Two values are equal") } else { console.log("Two values are not equal") ❌ } ``` ---- #### 避免混和型別的比較 ```diff= const x = 1 const s = "1" - if(x === s) { + if(x === Number(s)) { console.log("Two values are equal") ✅ } else { console.log("Two values are not equal") } ``` ---- #### 注意編譯器的警告 (DEMO) [![play-button](https://hackmd.io/_uploads/SydfgiNX0.png)](https://codepen.io/JimHu1492/pen/zYQGmZr?editors=1111) ---- <div style="display: flex; justify-content: center;"> <img src="https://hackmd.io/_uploads/r1J6PoV7R.png" width="50%" style="margin-right: 50px;background: white; padding: 10px" /> <div width="50%"> <h4>基本資料型態®</h4> <ul style="text-align: left; margin-left: -120px; font-size:24px; list-style: none"> <li>數值概論</li> <li style="color: #FCD354; list-style-image: url('https://hackmd.io/_uploads/Bk5lGONQC.png');background-size: 5px;"><b>整數</b></li> <li>浮點數</li> <li>字元和字串</li> <li>布林變數</li> <li>列舉型別</li> <li>具名常數</li> <li>陣列</li> <li>建立型別</li> <li>...</li> </ul> <img src="https://hackmd.io/_uploads/HkCAKsWQC.png" width="200px" style="position: absolute;right:20px;bottom: 10px;" /> </div> <div> ---- <h3 style="text-align:left;"><img src="https://hackmd.io/_uploads/HJMQOONXC.png" width="45px" style="margin-top: 3px;margin-right: 10px;"><span style="color: #FCD354;">整數</span></h3> <ul> <li>檢查整數除法(語言不同有不同結果)</li> <li>檢查整數溢出</li> <li>檢查中間結果溢出</li> </ul> ---- #### 檢查整數除法 ```python= # Python 範例 a = 7 b = 10 result = a // b # 整數除法 print(result) # 結果是 0 ``` ---- <table style="font-size:24px;"> <tr> <th>數據類型</th> <th>最小位元數 (Bits)</th> <th>範圍(假設符號)</th> </tr> <tr> <td>int</td> <td>16</td> <td>-32,768 到 32,767</td> </tr> <tr> <td>unsigned int</td> <td>16</td> <td>0 到 65,535</td> </tr> <tr> <td>short int</td> <td>16</td> <td>-32,768 到 32,767</td> </tr> <tr> <td>unsigned short int</td> <td>16</td> <td>0 到 65,535</td> </tr> <tr> <td>long int</td> <td>32</td> <td>-2,147,483,648 到 2,147,483,647</td> </tr> <tr> <td>unsigned long int</td> <td>32</td> <td>0 到 4,294,967,295</td> </tr> <tr> <td>long long int</td> <td>64</td> <td>-9,223,372,036,854,775,808 到 9,223,372,036,854,775,807</td> </tr> <tr> <td>unsigned long long int</td> <td>64</td> <td>0 到 18,446,744,073,709,551,615</td> </tr> </table> ---- #### 檢查整數溢出、檢查中間結果溢出 ``` function willAdditionOverflow(a, b) { if (typeof a !== 'number' || typeof b !== 'number') { throw new Error("Both arguments must be numbers"); } if (!Number.isSafeInteger(a) || !Number.isSafeInteger(b)) { return true; // 任何一個數值超出安全範圍 } let result = a + b; if (!Number.isSafeInteger(result)) { return true; // 結果超出安全範圍 } return false; } let a = Number.MAX_SAFE_INTEGER; let b = 1; if (willAdditionOverflow(a, b)) { console.log("Addition will cause overflow!"); } else { let result = a + b; console.log("Result: " + result); } ``` ---- <div style="display: flex; justify-content: center;"> <img src="https://upload.wikimedia.org/wikipedia/commons/4/4d/Float_mantissa_exponent.png" width="50%" style="margin-right: 50px;background: white; padding: 10px" /> <div width="50%"> <h4>基本資料型態®</h4> <ul style="text-align: left; margin-left: -120px; font-size:24px; list-style: none"> <li>數值概論</li> <li><b>整數</b></li> <li style="color: #FCD354; list-style-image: url('https://hackmd.io/_uploads/Bk5lGONQC.png');background-size: 5px;">浮點數</li> <li>字元和字串</li> <li>布林變數</li> <li>列舉型別</li> <li>具名常數</li> <li>陣列</li> <li>建立型別</li> <li>...</li> </ul> <img src="https://hackmd.io/_uploads/HkCAKsWQC.png" width="200px" style="position: absolute;right:20px;bottom: 10px;" /> </div> <div> ---- <h3 style="text-align:left;"><img src="https://hackmd.io/_uploads/HJMQOONXC.png" width="45px" style="margin-top: 3px;margin-right: 10px;"><span style="color: #FCD354;">浮點數</span></h3> <ul> <li>避免相差過大的數之間的加減運算(需排序處理)</li> <li>避免等量判斷</li> <li>處理捨入誤差問題</li> <li>檢查語言對特定資料型別的支援(VB, Currency 的型別)</li> </ul> ---- #### 避免相差過大的數之間的加減運算 ```python= # 排序 VS. 未排序 正確性比較 # 假設的浮點數數組 numbers = [10000000.0, 0.0001, 0.0002, 0.0003, 0.0004] # 從小到大排序 numbers_sorted = sorted(numbers) # 進行加法運算 total = 0 for number in numbers_sorted: total += number # 輸出結果 print("加總結果(排序後):", total) # 沒有排序直接加總,作為比較 total_unsorted = sum(numbers) print("加總結果(未排序):", total_unsorted) ``` ---- #### 避免等量判斷 ```python= # 將0.1相加10次 result = 0.1 * 10 nominal = 1.0 # 初始化結果變量 result = 0.0 # 使用for循環將0.1加10次 for _ in range(10): result += 0.1 # 輸出結果 print('結果數字為:', result) //結果數字為: 0.9999999999999999 print('nominal == result:', nominal == result) //nominal == result: False ``` ---- #### 避免等量判斷(解決辦法) - 使用精度範圍進行比較 - 使用 math.isclose 函數 - 使用固定點數表示法或高精度數學庫 [參考 chatGPT 解答](https://chat.openai.com/share/42dac35e-17bd-4eb1-90ac-63ea832ec76c) ---- #### 處理捨入誤差問題 - 換更高精準度的變數型別(同相差過大的數之間的加減運算問題) - 換成二進為編碼的十進位變數(BCD) ---- <iframe height="500" style="width: 100%;" scrolling="no" title="Untitled" src="https://codepen.io/JimHu1492/embed/PovYRvE?default-tab=" frameborder="no" loading="lazy" allowtransparency="true" allowfullscreen="true"> See the Pen <a href="https://codepen.io/JimHu1492/pen/PovYRvE"> Untitled</a> by Jim Hu (<a href="https://codepen.io/JimHu1492">@JimHu1492</a>) on <a href="https://codepen.io">CodePen</a>. </iframe> ---- #### 檢查語言對特定資料型別的支援 VB 有 Currency 這樣內建的型別支援 ---- <div style="display: flex; justify-content: center;"> <img src="https://hackmd.io/_uploads/r1ezlIPXA.png" width="50%" style="margin-right: 50px;background: white; padding: 10px" /> <div width="50%"> <h4>基本資料型態®</h4> <ul style="text-align: left; margin-left: -120px; font-size:24px; list-style: none"> <li>數值概論</li> <li><b>整數</b></li> <li>浮點數</li> <li style="color: #FCD354; list-style-image: url('https://hackmd.io/_uploads/Bk5lGONQC.png');background-size: 5px;">字元和字串</li> <li>布林變數</li> <li>列舉型別</li> <li>具名常數</li> <li>陣列</li> <li>建立型別</li> <li>...</li> </ul> <img src="https://hackmd.io/_uploads/HkCAKsWQC.png" width="200px" style="position: absolute;right:20px;bottom: 10px;" /> </div> <div> ---- <h3 style="text-align:left;"><img src="https://hackmd.io/_uploads/HJMQOONXC.png" width="45px" style="margin-top: 3px;margin-right: 10px;"><span style="color: #FCD354;">字元和字串</span></h3> <ul> <li>避免魔術字元和魔術字串</li> <li>避免 off-by-one 錯誤</li> <li>了解你的語言和開發環境是如何支援 Unicode 的</li> <li>在程式生命其中儘早決定國際化/本地化策略</li> <li>如果你知道只需要支持一種文字的語言,請考慮使用 ISO 8859 字元集</li> <li>如果你需要支援多種語言,請使用 Unicode 與 ISO 8859</li> <li>採用一致的字串型別轉換策略</li> </ul> ---- #### 避免魔術字元和魔術字串(Anti-pattern) ```C= #include <stdio.h> int main(void){ char i = 'A'; //宣告字元變數 i char j = 'z'; // 告字元變數 j printf("字元 A 的 ASCII 碼為 %d\n", (int)i); printf("字元 z 的 ASCII 碼為 %d\n", (int)j); return 0; } ``` ---- #### 避免魔術字元和魔術字串(修正) ``` #include <stdio.h> int main(void) { // 使用有意義的常數來替代魔術字元 const char UPPER_CASE_A = 'A'; // 宣告常數 UPPER_CASE_A const char LOWER_CASE_Z = 'z'; // 宣告常數 LOWER_CASE_Z printf("字元 %c 的 ASCII 碼為 %d\n", UPPER_CASE_A, (int)UPPER_CASE_A); printf("字元 %c 的 ASCII 碼為 %d\n", LOWER_CASE_Z, (int)LOWER_CASE_Z); return 0; } ``` ---- #### 避免魔術字元和魔術字串(output) ``` 字元 A 的 ASCII 碼為 65 字元 z 的 ASCII 碼為 122 ``` ---- #### (補)避免使用字面形式的字串的眾多原因如下: - 程式修改,只是修改字串容易造成錯誤 - 多國語系處理翻譯的字串應該存在放資源檔中而非程式變數 - 字串的字面表示形式通常占較多的儲存空間 - 字元和字串的字面表示形式的含意是模糊的(0x1B 使用 ESCAPE 表示) ---- #### (補)多國語系處理實例 ```JSON= { "app_name": "我的應用", "login": "登入", "description": "輸入您的使用者名稱和密碼以登入。" } ``` ---- #### (補)字元和字串的字面表示形式的含意是模糊的 0x1B, U+000A, &nbsp(No-Break Space) [參考資料](https://cloud.magiclen.org/tw/ascii) ---- #### 避免 off-by-one 錯誤 字串處理與陣列相同,要避免邊界取值的錯誤 ```python= words = ["hello", "world", "python", "programming"] # 假設我們想要打印每一個元素,但不小心設置了錯誤的迴圈條件 for i in range(len(words) + 1): # 錯誤:超出範圍 print(words[i]) ``` ---- #### 解你的語言和開發環境是如何支援 Unicode 的 <table style="font-size:24px; width:100%; border: 1px solid black;"> <tr> <th>語言</th> <th>字符表示</th> <th>字符串表示</th> <th>特點與支持</th> </tr> <tr> <td>C</td> <td>依賴實現</td> <td>依賴實現</td> <td>沒有內建 Unicode 支持<br>可使用第三方庫,如 ICU、libutf8<br>編譯器可能提供部分支持</td> </tr> <tr> <td>Java</td> <td>char (16 位 UTF-16)</td> <td>String (UTF-16)</td> <td>內建豐富的 Unicode 支持<br>自動處理多語言情境<br>提供豐富的字符串操作 API</td> </tr> <tr> <td>C#</td> <td>char (16 位 UTF-16)</td> <td>string (UTF-16)</td> <td>.NET 平台原生支持 Unicode<br>自動處理 Unicode 字符串<br>全面的 API 支持</td> </tr> <tr> <td>JavaScript</td> <td>UTF-16 代碼單元</td> <td>UTF-16 字符串</td> <td>ES6 增強 Unicode 支持<br>支持代理對字符<br>Unicode 正則表達式標誌 `u`</td> </tr> <tr> <td>Python</td> <td>Unicode 字符串</td> <td>Unicode 字符串</td> <td>Python 3 所有字符串為 Unicode<br>Python 2 需明確使用 `u'...'`<br>廣泛的 Unicode 操作支持</td> </tr> </table> ---- - 在程式生命其中儘早決定國際化/本地化策略 - 如果你知道只需要支持一種文字的語言,請考慮使用 ISO 8859 字元集 - 如果你需要支援多種語言,請使用 Unicode 與 ISO 8859 ---- <table style="font-size:24px; width:100%; border: 1px solid black;"> <tr> <th>特性</th> <th>Unicode</th> <th>ISO 8859</th> </tr> <tr> <td>支持的語言範圍</td> <td>全球範圍內的幾乎所有語言</td> <td>限定於特定的西歐、中歐等語言群</td> </tr> <tr> <td>字符大小</td> <td>變長(通常為 1 至 4 位元組)</td> <td>固定大小(1 位元組)</td> </tr> <tr> <td>編碼方式</td> <td>多種(如 UTF-8, UTF-16, UTF-32 等)</td> <td>單一(每種 ISO 8859 系列有其編碼方式)</td> </tr> <tr> <td>設計目的</td> <td>為了統一全球範圍內多語言的字符表示</td> <td>為了擴展 ASCII 並支持特定語言的字符集</td> </tr> <tr> <td>兼容性</td> <td>高兼容性,被現代操作系統、瀏覽器和應用廣泛支持</td> <td>兼容性相對較低,現代應用中逐漸被 Unicode 取代</td> </tr> <tr> <td>應用場景</td> <td>適用於需要支持多語言的現代軟件和網路應用</td> <td>適用於只需支持單一語言或特定語言群的較舊系統和應用</td> </tr> </table> ---- ```python= # Python 3 Unicode 字串 s = "Hello, world!" # 自動處理為 Unicode 字串 # Python 3 字節串 b = b"Hello, bytes!" # 轉換 Unicode 字串為字節串 encoded = s.encode('utf-8') # 從 Unicode 編碼為 UTF-8 字節串 # 轉換字節串為 Unicode 字串 decoded = encoded.decode('utf-8') # 從 UTF-8 字節串解碼為 Unicode 字串 ``` ---- #### 採用一致的字串型別轉換策略 1. 保存同一種格式 2. 在輸入輸出中操作轉換為其他格式 ```python= # 從外部來源讀取時,確保使用適當的編碼將字節數據轉換為 Unicode 字串 # 例如,從文件讀取 with open('example.txt', 'r', encoding='utf-8') as file: text = file.read() # text 是 Unicode 字串 # 來自網絡的數據 # 假設 response 是一個 HTTP 響應對象 data = response.content.decode('utf-8') # 將字節數據解碼為 Unicode 字串 ``` ---- #### (補)C 語言中的字串 - 注意字串指標和字元陣列之間的差異 - 把 C 風格字串的長度宣告為 CONSTANT + 1 - 用 null 初始化字串以避免沒有終端的字串 - 用字串陣列取代 C 中的指標 - 用 strncpy()取代 strcpy()以避免無終端的字串 ---- #### (補)注意字串指標和字元陣列之間的差異 ```C= char strArray[6] = "hello"; // 儲存了 'h', 'e', 'l', 'l', 'o', '\0' char *strPtr = "hello"; // 指向 "hello" 字串常數的指標 ``` ---- #### (補)把 C 風格字串的長度宣告為 CONSTANT + 1 ```C= char greeting[8] = "Hello!"; // "Hello!" 需要 7 個字元空間 + 1 個 '\0' ``` ---- #### (補)用 null 初始化字串以避免沒有終端的字串 ```C= char buffer[100] = {0}; // 所有字元都初始化為 '\0' ``` ---- #### (補)用字串陣列取代 C 中的指標 ```C= char *names[] = {"Alice", "Bob", "Charlie", "David", "Eve"}; char names[5][10] = { "Alice", "Bob", "Charlie", "David", "Eve" }; ``` ---- #### (補)用 strncpy()取代 strcpy()以避免無終端的字串 ```C= char src[] = "Hello World"; char dest[6]; strncpy(dest, src, 5); dest[5] = '\0'; // 確保有終止符號 ``` ---- <div style="display: flex; justify-content: center;"> <img src="https://www.freecodecamp.org/news/content/images/2020/01/image-2.png" width="50%" style="margin-right: 50px;background: white; padding: 10px" /> <div width="50%"> <h4>基本資料型態®</h4> <ul style="text-align: left; margin-left: -120px; font-size:24px; list-style: none"> <li>數值概論</li> <li><b>整數</b></li> <li>浮點數</li> <li>字元和字串</li> <li style="color: #FCD354; list-style-image: url('https://hackmd.io/_uploads/Bk5lGONQC.png');background-size: 5px;">布林變數</li> <li>列舉型別</li> <li>具名常數</li> <li>陣列</li> <li>建立型別</li> <li>...</li> </ul> <img src="https://hackmd.io/_uploads/HkCAKsWQC.png" width="200px" style="position: absolute;right:20px;bottom: 10px;" /> </div> <div> ---- <h3 style="text-align:left;"><img src="https://hackmd.io/_uploads/HJMQOONXC.png" width="45px" style="margin-top: 3px;margin-right: 10px;"><span style="color: #FCD354;">布林變數</span></h3> <ul> <li>用布林變數對程式加以文件說明</li> <li>用布林變數來簡化複雜的判斷</li> <li>如果需要的話,建立你自己的布林型別</li> </ul> ---- #### 用布林變數對程式加以文件說明 - 可以把判斷用變數存取,以變數命名的方式增加程式碼可讀性 - 可以使用函式的方式 ---- #### 用布林變數對程式加以文件說明 ``` def is_student_eligible(scores, parent_is_alumni): # scores 是一個字典,包含學生的各科成績 # parent_is_alumni 是一個布林值,表示至少有一位父母是校友 # 使用列表推導式來計算超過 60 分的科目數量 passing_count = sum(1 for score in scores.values() if score > 60) # 判斷是否符合入學資格 if (parent_is_alumni and passing_count >= 2) or passing_count >= 3: return True # 符合入學資格 else: return False # 不符合入學資格 # 測試代碼 scores_example = {'國語': 55, '英語': 65, '數學': 70, '物理': 55, '化學': 58} parent_is_alumni_example = True if (is_student_eligible(scores_example, parent_is_alumni_example)) print("符合") else: print("不符合") ``` https://www.online-python.com/QCD7RkbuPj ---- #### 如果需要的話,建立你自己的布林型別 ```C= typedef int BOOLEAN; ``` ```typescript= type IsActive = boolean; function setActive(status: IsActive) { // 做一些事情 } ``` ---- <div style="display: flex; justify-content: center;"> <img src="https://i.ytimg.com/vi/5VyDsO0mFDU/maxresdefault.jpg" width="50%" style="margin-right: 50px;background: white; padding: 10px" /> <div width="50%"> <h4>基本資料型態®</h4> <ul style="text-align: left; margin-left: -120px; font-size:24px; list-style: none"> <li>數值概論</li> <li><b>整數</b></li> <li>浮點數</li> <li>字元和字串</li> <li>布林變數</li> <li style="color: #FCD354; list-style-image: url('https://hackmd.io/_uploads/Bk5lGONQC.png');background-size: 5px;">列舉型別</li> <li>具名常數</li> <li>陣列</li> <li>建立型別</li> <li>...</li> </ul> <img src="https://hackmd.io/_uploads/HkCAKsWQC.png" width="200px" style="position: absolute;right:20px;bottom: 10px;" /> </div> <div> ---- <h3 style="text-align:left;"><img src="https://hackmd.io/_uploads/HJMQOONXC.png" width="45px" style="margin-top: 3px;margin-right: 10px;"><span style="color: #FCD354;">列舉型別</span></h3> <ul> <li>用列舉來提高可靠性</li> <li>用列舉型別來簡化修改</li> <li>將列舉型別作為布林變數的替代方案</li> <li>檢查非法數值</li> <li>定義出列舉的第一項和最後一項,以便用於迴圈限制</li> <li>把列舉型別的第一個元素留做非法值</li> <li>明確定義專案程式碼編寫標準中第一個和最後一個元素的使用規則,並且在使用時保持一致</li> <li>警惕給列舉元素明確賦值帶來的錯誤</li> <li>如果你的語言裡沒有列舉型別</li> </ul> ---- #### (補)用列舉來提高可靠性 物件與列舉差異 <table style="font-size:16px; width:100%; border-collapse: collapse; border: 1px solid black;"> <tr> <th style="border: 1px solid black; padding: 8px;">特性</th> <th style="border: 1px solid black; padding: 8px;">列舉 (Enum)</th> <th style="border: 1px solid black; padding: 8px;">物件 (Object)</th> </tr> <tr> <td style="border: 1px solid black; padding: 8px;">定義</td> <td style="border: 1px solid black; padding: 8px;">資料型別,包含一組命名的整數常量。</td> <td style="border: 1px solid black; padding: 8px;">結構,可以包含多個值(屬性)和方法。</td> </tr> <tr> <td style="border: 1px solid black; padding: 8px;">用途</td> <td style="border: 1px solid black; padding: 8px;">用於定義一組相關的常量,提高代碼清晰度。</td> <td style="border: 1px solid black; padding: 8px;">用於封裝和模型化複雜的數據和行為,使代碼易於理解、擴展。</td> </tr> <tr> <td style="border: 1px solid black; padding: 8px;">特性</td> <td style="border: 1px solid black; padding: 8px;">有限的值集合,整數映射。</td> <td style="border: 1px solid black; padding: 8px;">靈活性和動態性,支持繼承和多態性。</td> </tr> <tr> <td style="border: 1px solid black; padding: 8px;">表示方式</td> <td style="border: 1px solid black; padding: 8px;">簡單,不含方法。</td> <td style="border: 1px solid black; padding: 8px;">複雜,包含數據和行為。</td> </tr> </table> ---- #### 用列舉提高可靠性 ```Ada= -- 定義一個列舉型別 Traffic_Light type Traffic_Light is (Red, Yellow, Green); -- 宣告一個變數來儲存交通燈狀態 signal : Traffic_Light; -- 一個過程來模擬交通燈變化 procedure Switch_Light(S : in out Traffic_Light) is begin case S is when Red => -- 如果燈是紅色,下一個應該是綠色 S := Green; when Yellow => -- 如果燈是黃色,下一個應該是紅色 S := Red; when Green => -- 如果燈是綠色,下一個應該是黃色 S := Yellow; end case; end Switch_Light; -- 使用列舉的例子 begin -- 初始化信號為紅燈 signal := Red; -- 印出當前燈號 Ada.Text_IO.Put_Line("Current Light: " & Traffic_Light'Image(signal)); -- 變換信號 Switch_Light(signal); -- 再次印出變換後的燈號 Ada.Text_IO.Put_Line("Next Light: " & Traffic_Light'Image(signal)); end; ``` ---- #### 用列舉來簡化修改 - yello: 1 - red: 2 - green: 3 => 4 共用列舉只要修改列舉就能全部修改定義 ---- #### 將列舉型別作為布林變數的替代方案 ``` from enum import Enum # 定義一個列舉型別 ApiResponseStatus class ApiResponseStatus(Enum): SUCCESS = 200 NOT_FOUND = 404 BAD_REQUEST = 400 UNAUTHORIZED = 401 FORBIDDEN = 403 SERVER_ERROR = 500 # 這裡是一個模擬的 API 回應函數 def mock_api_call(): # 模擬一個錯誤回應,例如 401 Unauthorized return ApiResponseStatus.UNAUTHORIZED # 使用例子 response_status = mock_api_call() if response_status != ApiResponseStatus.SUCCESS: print(f"Error occurred: {response_status.name}") else: print("API call successful!") ``` ---- #### 檢查非法數值 ``` from enum import Enum # 定義一個列舉型別 ApiResponseStatus class ApiResponseStatus(Enum): SUCCESS = 200 NOT_FOUND = 404 BAD_REQUEST = 400 UNAUTHORIZED = 401 FORBIDDEN = 403 SERVER_ERROR = 500 # 模擬一個 API 回應函數 def mock_api_call(): # 模擬一個未定義的錯誤狀態,例如 418 I'm a teapot return 418 # 檢查回應是否在 ApiResponseStatus 中 def check_response(status_code): for status in ApiResponseStatus: if status.value == status_code: return status # 如果未找到匹配的列舉項,則返回 None return None # 使用例子 response_code = mock_api_call() response_status = check_response(response_code) if response_status is None: print(f"Unknown error occurred with status code: {response_code}") elif response_status == ApiResponseStatus.SUCCESS: print("API call successful!") else: print(f"Error occurred: {response_status.name}") ``` ---- #### 定義出列舉第一項和最後一項,以便用於迴圈限制 ``` from enum import Enum class Weekday(Enum): FIRST = 0 # 用來標記第一筆,非具體一天 MONDAY = 1 TUESDAY = 2 WEDNESDAY = 3 THURSDAY = 4 FRIDAY = 5 SATURDAY = 6 SUNDAY = 7 LAST = 8 # 用來標記最後一筆,非具體一天 # 遍歷 Weekday 中介於 FIRST 和 LAST 之間的值 for day in Weekday: if Weekday.FIRST.value < day.value < Weekday.LAST.value: print(day.name) ``` ---- #### 把列舉型別的第一個元素留做非法值 ``` from enum import Enum class StatusCode(Enum): INVALID = 0 # 非法或無效的狀態 OK = 1 ERROR = 2 NOT_FOUND = 3 # 函數來模擬 API 請求 def mock_api_call() -> StatusCode: # 模擬一個 API 請求,假設這個請求失敗 return StatusCode.ERROR # 使用列舉的例子 status = mock_api_call() if status == StatusCode.INVALID: print("Error: Received an invalid status code.") elif status == StatusCode.OK: print("API call successful!") else: print(f"API call resulted in an error with status: {status.name}") ``` ---- #### 明確定義專案程式碼編寫中第一個和最後一個元素的使用規則 ``` from enum import Enum, auto class ConfigOption(Enum): INVALID = auto() # 第一個元素,用於無效值 AUDIO = auto() VIDEO = auto() NETWORK = auto() LAST = auto() # 最後一個元素,用於界定範圍 # 函數來處理不同的配置選項 def process_config(option: ConfigOption): if option == ConfigOption.INVALID: print("Error: Invalid configuration option selected.") return print(f"Processing {option.name} configuration...") # 遍歷 ConfigOption,略過 INVALID 和 LAST for option in ConfigOption: if option not in (ConfigOption.INVALID, ConfigOption.LAST): process_config(option) ``` ---- #### 如果你的語言裡沒有列舉型別 1. 定義常量 2. 物件封裝 3. 類別和靜態屬性 4. 使用陣列或列表 ---- ##### 物件封裝 ```javascript= const StatusCode = Object.freeze({ INVALID: 0, OK: 1, ERROR: 2, NOT_FOUND: 3 }); ``` ---- ##### 類別和靜態屬性 ```python= class StatusCode: INVALID = 0 OK = 1 ERROR = 2 NOT_FOUND = 3 ``` ---- <div style="display: flex; justify-content: center;"> <img src="https://slidesplayer.com/slide/16398594/95/images/116/%E5%85%B7%E5%90%8D%E5%B8%B8%E6%95%B8+%28Named+Constant%29.jpg" width="50%" style="margin-right: 50px;background: white; padding: 10px" /> <div width="50%"> <h4>基本資料型態®</h4> <ul style="text-align: left; margin-left: -120px; font-size:24px; list-style: none"> <li>數值概論</li> <li><b>整數</b></li> <li>浮點數</li> <li>字元和字串</li> <li>布林變數</li> <li>列舉型別</li> <li style="color: #FCD354; list-style-image: url('https://hackmd.io/_uploads/Bk5lGONQC.png');background-size: 5px;">具名常數</li> <li>陣列</li> <li>建立型別</li> <li>...</li> </ul> <img src="https://hackmd.io/_uploads/HkCAKsWQC.png" width="200px" style="position: absolute;right:20px;bottom: 10px;" /> </div> <div> ---- <h3 style="text-align:left;"><img src="https://hackmd.io/_uploads/HJMQOONXC.png" width="45px" style="margin-top: 3px;margin-right: 10px;"><span style="color: #FCD354;">具名常數</span></h3> <ul> <li>在資料宣告中使用具名常數</li> <li>避免使用字面常數,即使是「安全」的</li> <li>用具有適當作用範圍的變數或類別來模擬具名常數</li> <li>統一使用具名常數</li> </ul> ---- #### 在資料宣告中使用具名常數 ```python= class GameSettings: MAX_PLAYERS = 10 MIN_PLAYERS = 2 # 使用具名常數來宣告遊戲的最大玩家數 max_players_allowed = GameSettings.MAX_PLAYERS print(f"Maximum players allowed: {max_players_allowed}") ``` ---- #### 避免使用字面常數,即使是「安全」的 ```javascript= class HTTP { static TIMEOUT = 5000; // 5000 毫秒的超時時間 } // 使用具名常數來設定請求超時,避免直接使用字面常數 fetch('https://api.example.com/data', { timeout: HTTP.TIMEOUT }) .then(response => console.log('Request succeeded')) .catch(error => console.log('Request failed with timeout')); ``` ---- #### 用具有適當作用範圍的變數或類別來模擬具名常數 ```Java= public class Configuration { public static final int DEFAULT_PORT = 8080; } // 在伺服器設定中使用具有適當作用範圍的具名常數 int serverPort = Configuration.DEFAULT_PORT; System.out.println("Server running on port: " + serverPort); ``` ---- #### 統一使用具名常數 ```C# public class Constants { public const string DB_CONNECTION_STRING = "Server=myServerAddress;Database=myDataBase;User Id=myUsername;Password=myPassword;"; } // 使用具名常數來統一資料庫連接字串 string connectionString = Constants.DB_CONNECTION_STRING; ``` ---- <div style="display: flex; justify-content: center;"> <img src="https://programming.im.ncnu.edu.tw/array.jpg" width="50%" style="margin-right: 50px;background: white; padding: 10px" /> <div width="50%"> <h4>基本資料型態®</h4> <ul style="text-align: left; margin-left: -120px; font-size:24px; list-style: none"> <li>數值概論</li> <li><b>整數</b></li> <li>浮點數</li> <li>字元和字串</li> <li>布林變數</li> <li>列舉型別</li> <li>具名常數</li> <li style="color: #FCD354; list-style-image: url('https://hackmd.io/_uploads/Bk5lGONQC.png');background-size: 5px;">陣列</li> <li>建立型別</li> <li>...</li> </ul> <img src="https://hackmd.io/_uploads/HkCAKsWQC.png" width="200px" style="position: absolute;right:20px;bottom: 10px;" /> </div> <div> ---- <h3 style="text-align:left;"><img src="https://hackmd.io/_uploads/HJMQOONXC.png" width="45px" style="margin-top: 3px;margin-right: 10px;"><span style="color: #FCD354;">陣列</span></h3> <ul> <li>確認所有的陣列指標都沒有超出陣列的邊界</li> <li>考慮用容器來取代陣列,或者將陣列作為順序化結構來處理</li> <li>檢查陣列的邊界點</li> <li>如果陣列是多維的,確認指標的使用順序是正確的</li> <li>提防指標交互干擾</li> </ul> ---- #### 確認所有的陣列指標都沒有超出陣列的邊界 ``` numbers = [1, 2, 3, 4, 5] index = 5 # 這個索引是超出範圍的,因為最大索引應該是 4 if index < len(numbers): print(numbers[index]) else: print("Index is out of bounds.") ``` ---- #### 考慮用容器來取代陣列,或者將陣列作為順序化結構來處理 ``` # 陣列容器相對於 Stack, Queue, Map, Set 等不同的容器,具有順序化的特色, 應該藉由順序化結構來處理。(討論後補充) items = [] for i in range(10): # 動態添加元素 items.append(i ** 2) print(items) ``` ---- #### 檢查陣列的邊界點 ``` # 檢查列表的首尾元素 numbers = [1, 2, 3, 4, 5] first = numbers[0] last = numbers[-1] # Python 允許使用負索引來從列表末尾訪問元素 print(f"First: {first}, Last: {last}") ``` ---- #### 如果陣列是多維的,確認指標的使用順序是正確的 ``` # 多維列表 matrix = [[1, 2, 3], [4, 5, 6], [7, 8, 9]] for row in matrix: for item in row: print(item, end=' ') print() ``` ---- #### 提防指標交互干擾 ``` # 指標交互干擾的例子 original = [1, 2, 3] copy = original # 這裡不是創建一個新的列表,而是引用同一個列表 copy[0] = 99 # 修改 copy 會影響 original print(f"Original: {original}") ``` ---- <div style="display: flex; justify-content: center;"> <img src="" width="50%" style="margin-right: 50px;background: white; padding: 10px" /> <div width="50%"> <h4>基本資料型態®</h4> <ul style="text-align: left; margin-left: -120px; font-size:24px; list-style: none"> <li>數值概論</li> <li><b>整數</b></li> <li>浮點數</li> <li>字元和字串</li> <li>布林變數</li> <li>列舉型別</li> <li>具名常數</li> <li>陣列</li> <li style="color: #FCD354; list-style-image: url('https://hackmd.io/_uploads/Bk5lGONQC.png');background-size: 5px;">建立型別</li> <li>...</li> </ul> <img src="https://hackmd.io/_uploads/HkCAKsWQC.png" width="200px" style="position: absolute;right:20px;bottom: 10px;" /> </div> <div> ---- #### 建立型別實例(資料定義有跨型別的需要?) ```C++= typedef float Coordinate; // for coordinate variables Routine1( ... ) { Coordinate latitude; Coordinate longitude; Coordinate elevation; ... } Routine2( ... ) { Coordinate x; Coordinate y; Coordinate z; ... } ``` ---- #### 建立自己型別的原因 - 易於修改 - 避免過多的資訊分發 //Const NAME_LENGTH = 30; - 增加可靠性 //Ada type Age is range 0..99 - 彌補語言的不足 // typedef in t Boolean; ---- #### 建立自訂義資料型別的指導原則 - 給所建立的型別取功能導向的名稱 - 避免使用預定義型別 - 不要重訂義一個預定義的型別 - 定義替代型別以便於移植 - 考慮建立一個類別而不是使用 typedef ---- 其他: ---- #### (補充)「靜態型別 vs. 動態型別」與「強型別 vs. 弱型別」 [![12_type_system](https://hackmd.io/_uploads/HJWUlh-mC.jpg)](https://blog.tarswork.com/post/programming-language-type-system/) ---- #### 思考:自定義型別如何管理? ----
{"title":"基本資料型別®","description":"12.1 數值概論12.2 整數12.3 浮點數12.4 字元和字串12.5 布林變數12.6 列舉型別12.7 具名常數12.8 陣列12.9 建立你自己的型別(型別別名)要點","contributors":"[{\"id\":\"166114a0-368f-4432-b369-5d1bdfcf0880\",\"add\":70593,\"del\":30665}]"}
    187 views