沒有測試的正規(guī)表示式是自找麻煩 - 不要偷懶。它是免費的人工智慧!
TL;DR:使用清晰簡潔的正規(guī)表示式,並徹底測試它們。
問題
- 可讀性
- 沒有測試用例
- 錯過了邊緣狀況
- 調(diào)試挑戰(zhàn)
- 不清楚的失敗
- 隱藏缺陷
解決方案
- 讓你最喜歡的人工智慧寫測試案例
- 將複雜的正規(guī)表示式分解為更小、更易讀的部分。
- 檢查邊緣狀況
- 驗證輸出
- 建立測試後重構(gòu)正規(guī)表示式
- 改進(jìn)錯誤訊息
情境
正規(guī)表示式很強(qiáng)大,但也很棘手。
如果您在沒有測試的情況下編寫正規(guī)表示式,則會出現(xiàn)意外錯誤。
如果您編寫神秘的正規(guī)表示式並跳過自動化測試,您可能會錯過重要的案例,從而導(dǎo)致安全問題或使用者沮喪。
範(fàn)例程式碼
錯誤的
public class PasswordValidator { public static boolean isValidPassword(String password) { return password.matches( "^(?=.*[a-z])(?=.*[A-Z])(?=.*\d)[a-zA-Z\d]{8,}$"); // This is a cryptic Regular Expression } }
正確的
import java.util.ArrayList; import java.util.List; public class PasswordValidator { public static List<String> validatePassword(String password) { List<String> errors = new ArrayList<>(); if (password.length() < 8) { errors.add( "Password must be at least 8 characters long."); } if (!password.matches(".*[A-Z].*")) { errors.add( "Password must contain at least one uppercase letter."); } if (!password.matches(".*[a-z].*")) { errors.add( "Password must contain at least one lowercase letter."); } if (!password.matches(".*\d.*")) { errors.add( "Password must contain at least one digit."); } if (errors.isEmpty()) { errors.add( "Password is valid."); } return errors; // You no longer need a Regular Expression!! } } import static org.junit.Assert.*; import org.junit.Test; public class PasswordValidatorTest { // Now you have a lot of tests // You can use a Regular Expression, // a String Validator // an External Library // Whatever you want as long as it passes the tests! @Test public void testValidPassword() { List<String> result = PasswordValidator.validatePassword( "StrongPass1"); assertEquals("Password is valid.", result.get(0)); } @Test public void testTooShortPassword() { List<String> result = PasswordValidator.validatePassword( "Short1"); assertTrue(result.contains( "Password must be at least 8 characters long.")); } @Test public void testNoUppercase() { List<String> result = PasswordValidator.validatePassword( "nouppercase1"); assertTrue( result.contains( "Password must contain at least one uppercase letter.")); } @Test public void testNoLowercase() { List<String> result = PasswordValidator.validatePassword( "NOLOWERCASE1"); assertTrue(result.contains( "Password must contain at least one lowercase letter.")); } @Test public void testNoNumber() { List<String> result = PasswordValidator.validatePassword( "NoNumberPass"); assertTrue(result.contains( "Password must contain at least one digit.")); } }
偵測
[X] 自動
您可以透過將正規(guī)表示式變更為失敗並執(zhí)行所有測試來偵測它何時被發(fā)現(xiàn)。
如果您的驗證傳回「false」而沒有使用者友善的解釋,則明確表明您需要重構(gòu)它並改善回饋。
標(biāo)籤
- 測試
等級
[X] 初學(xué)者
人工智慧世代
人工智慧可以產(chǎn)生正規(guī)表示式,但通常無法提供有用的錯誤訊息。
如果沒有適當(dāng)?shù)恼f明,人工智慧產(chǎn)生的驗證器可能無法指導(dǎo)使用者修復(fù)他們的輸入。
人工智慧檢測
人工智慧可以偵測基本的正規(guī)表示式模式和缺失的回饋,並提供清晰的提示。
除非特別要求,否則它可能不會自動建立詳細(xì)的測試案例或描述。
嘗試一下!
記?。喝斯ぶ腔壑頃负芏噱e誤
Without Proper Instructions | With Specific Instructions |
---|---|
ChatGPT | ChatGPT |
Claude | Claude |
Perplexity | Perplexity |
Copilot | Copilot |
Gemini | Gemini |
結(jié)論
沒有明確回饋的正規(guī)表示式對使用者不友善且容易出錯。
如果您描述了它們失敗的原因並編寫了徹底的測試以確保您的正規(guī)表示式按預(yù)期工作,這將會有所幫助。
關(guān)係

Code Smell 41 - 正規(guī)表示式濫用者
馬克西·孔蒂耶里 ?2020 年 12 月 3 日

Code Smell 185 - 邪惡的正規(guī)表示式
馬克西·孔蒂耶里 ?22 年 12 月 8 日

Code Smell 97 - 沒有同理心的錯誤訊息
馬克西·孔蒂耶里 ?2021 年 10 月 27 日
免責(zé)聲明
程式碼味道是我的觀點。
製作人員
照片由 Unsplash 上的 rc.xyz NFT 圖庫拍攝
回饋是冠軍的早餐。
肯·布蘭查德

軟體工程精彩名言
馬克西·孔蒂耶里 ?2020 年 12 月 28 日
本文是 CodeSmell 系列的一部分。

如何找出程式碼中的臭部分
馬克西·孔蒂耶里 ? 21 年 5 月 21 日
以上是程式碼異味 - 未經(jīng)測試的正規(guī)表示式的詳細(xì)內(nèi)容。更多資訊請關(guān)注PHP中文網(wǎng)其他相關(guān)文章!

熱AI工具

Undress AI Tool
免費脫衣圖片

Undresser.AI Undress
人工智慧驅(qū)動的應(yīng)用程序,用於創(chuàng)建逼真的裸體照片

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

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

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

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強(qiáng)大的PHP整合開發(fā)環(huán)境

Dreamweaver CS6
視覺化網(wǎng)頁開發(fā)工具

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

Callable和Runnable在Java中主要有三點區(qū)別。第一,Callable的call()方法可以返回結(jié)果,適合需要返回值的任務(wù),如Callable;而Runnable的run()方法無返回值,適用於無需返回的任務(wù),如日誌記錄。第二,Callable允許拋出checked異常,便於錯誤傳遞;而Runnable必須在內(nèi)部處理異常。第三,Runnable可直接傳給Thread或ExecutorService,而Callable只能提交給ExecutorService,並返回Future對像以

Java支持異步編程的方式包括使用CompletableFuture、響應(yīng)式流(如ProjectReactor)以及Java19 中的虛擬線程。 1.CompletableFuture通過鍊式調(diào)用提升代碼可讀性和維護(hù)性,支持任務(wù)編排和異常處理;2.ProjectReactor提供Mono和Flux類型實現(xiàn)響應(yīng)式編程,具備背壓機(jī)制和豐富的操作符;3.虛擬線程減少並發(fā)成本,適用於I/O密集型任務(wù),與傳統(tǒng)平臺線程相比更輕量且易於擴(kuò)展。每種方式均有適用場景,應(yīng)根據(jù)需求選擇合適工具並避免混合模型以保持簡潔性

在Java中,枚舉(enum)適合表示固定常量集合,最佳實踐包括:1.用enum表示固定狀態(tài)或選項,提升類型安全和可讀性;2.為枚舉添加屬性和方法以增強(qiáng)靈活性,如定義字段、構(gòu)造函數(shù)、輔助方法等;3.使用EnumMap和EnumSet提高性能和類型安全性,因其基於數(shù)組實現(xiàn)更高效;4.避免濫用enum,如動態(tài)值、頻繁變更或複雜邏輯場景應(yīng)使用其他方式替代。正確使用enum能提升代碼質(zhì)量並減少錯誤,但需注意其適用邊界。

JavaNIO是Java1.4引入的新型IOAPI,1)面向緩衝區(qū)和通道,2)包含Buffer、Channel和Selector核心組件,3)支持非阻塞模式,4)相比傳統(tǒng)IO更高效處理並發(fā)連接。其優(yōu)勢體現(xiàn)在:1)非阻塞IO減少線程開銷,2)Buffer提升數(shù)據(jù)傳輸效率,3)Selector實現(xiàn)多路復(fù)用,4)內(nèi)存映射加快文件讀寫。使用時需注意:1)Buffer的flip/clear操作易混淆,2)非阻塞下需手動處理不完整數(shù)據(jù),3)Selector註冊需及時取消,4)NIO並非適用於所有場景。

Java的類加載機(jī)制通過ClassLoader實現(xiàn),其核心工作流程分為加載、鏈接和初始化三個階段。加載階段由ClassLoader動態(tài)讀取類的字節(jié)碼並創(chuàng)建Class對象;鏈接包括驗證類的正確性、為靜態(tài)變量分配內(nèi)存及解析符號引用;初始化則執(zhí)行靜態(tài)代碼塊和靜態(tài)變量賦值。類加載採用雙親委派模型,優(yōu)先委託父類加載器查找類,依次嘗試Bootstrap、Extension和ApplicationClassLoader,確保核心類庫安全且避免重複加載。開發(fā)者可自定義ClassLoader,如URLClassL

Javaprovidesmultiplesynchronizationtoolsforthreadsafety.1.synchronizedblocksensuremutualexclusionbylockingmethodsorspecificcodesections.2.ReentrantLockoffersadvancedcontrol,includingtryLockandfairnesspolicies.3.Conditionvariablesallowthreadstowaitfor

Java異常處理的關(guān)鍵在於區(qū)分checked和unchecked異常並合理使用try-catch、finally及日誌記錄。 1.checked異常如IOException需強(qiáng)制處理,適用於可預(yù)期的外部問題;2.unchecked異常如NullPointerException通常由程序邏輯錯誤引起,屬於運(yùn)行時錯誤;3.捕獲異常時應(yīng)具體明確,避免籠統(tǒng)捕獲Exception;4.推薦使用try-with-resources自動關(guān)閉資源,減少手動清理代碼;5.異常處理中應(yīng)結(jié)合日誌框架記錄詳細(xì)信息,便於後

HashMap在Java中通過哈希表實現(xiàn)鍵值對存儲,其核心在於快速定位數(shù)據(jù)位置。 1.首先使用鍵的hashCode()方法生成哈希值,並通過位運(yùn)算轉(zhuǎn)換為數(shù)組索引;2.不同對象可能產(chǎn)生相同哈希值,導(dǎo)致衝突,此時以鍊錶形式掛載節(jié)點,JDK8後鍊錶過長(默認(rèn)長度8)則轉(zhuǎn)為紅黑樹提升效率;3.使用自定義類作鍵時必須重寫equals()和hashCode()方法;4.HashMap動態(tài)擴(kuò)容,當(dāng)元素數(shù)超過容量乘以負(fù)載因子(默認(rèn)0.75)時,擴(kuò)容並重新哈希;5.HashMap非線程安全,多線程下應(yīng)使用Concu
