揭露 Go 中隱藏的測試陷阱:避免誤報
測驗中的惡夢將是誤報。 「一切都會過去!驚人的!」直到未來的某個未知時間,所有地雷一起爆炸,將你的團隊炸入地獄。
測試可能默默失敗的原因有很多。
今天我要講一個很基本的原因:不知道哪些是測驗。
為什麼你不知道哪些是測驗?
大多數人都是半途而廢地加入 Go 專案的。大多數人透過在現實生活中使用語言來學習語言。
因此,當有人用testify這樣的測試框架來建立專案時,你很可能會認為下面這樣的方法就是測試。
func (suite *ExampleTestSuite) TestExample() { suite.Equal(5, suite.VariableThatShouldStartAtFive) }
然後您加入另一種方法,例如 TestAnotherCase 並發現它有效。您認為您非常清楚什麼是測試。
測試在不同的框架中有不同的意義
您所說的「測試」可能與 Go 套件所說的測試不同。
從內建的測試包中,測試可以是以下形式的任何函數
func TestXxx(*testing.T)
當然,由於內建的測試包功能有限,大多數專案都使用 testify/suite 或其他類似的第三方套件作為測試框架。從 testify/suite 的角度來看,什麼是測試?
加入任何以「Test」開頭的方法來新增測試
看,我們對測驗有兩種不同的定義。
使用第三方測試工具時就會出現問題
使用mockery等工具時,你會閱讀以下內容
您不必再擔心忘記 AssertExpectations 方法呼叫...AssertExpectations 方法已註冊為在測試結束時呼叫
太棒了! 「所以我只需要創建一個模擬,當預期的行為發生時,包就會通知我」。
那就是陷阱。
當mockery在測試結束時說時,它實際上意味著來自testing的定義,而不是來自testify/suite的定義。
因此,當您有以下程式碼時,您將看到 TestA 和 TestB 都通過,即使它們都應該失敗,因為 TestA 中的模擬設定在 TestB 中使用。
package mockandsubtest import ( "fmt" "testing" "github.com/stretchr/testify/suite" ) // Prod code type ExternalService interface { Work() } type Server struct { externalService ExternalService } func NewServer(externalService ExternalService) *Server { return &Server{ externalService: externalService, } } // Test code type ServerSuite struct { suite.Suite ExternalService *MockExternalService Server } func TestServerSuite(t *testing.T) { suite.Run(t, &ServerSuite{}) } // Run before all test cases func (s *ServerSuite) SetupSuite() { s.ExternalService = NewMockExternalService(s.T()) s.Server = Server{externalService: s.ExternalService} } // In this test, Work is set up to be called once but not called func (s *ServerSuite) TestA() { fmt.Println("TestA is running") s.ExternalService.EXPECT().Work().Times(1) } // In this test, Work is called once unexpectedly func (s *ServerSuite) TestB() { fmt.Println("TestB is running") s.Server.externalService.Work() }
運行上述程式碼的結果是
TestA is running TestB is running PASS
解釋
事實證明,從測驗和嘲笑的角度來看,只有 TestServerSuite 才被視為測驗。這就是為什麼在 TestServerSuite 末尾呼叫 AssertExpectations 的原因,即使 TestA 和 TestB 是由 testify/suite 內部執行的。
從mockery的角度來看,s.ExternalService預計會被呼叫一次,並且在TestServerSuite的生命週期中實際上會被呼叫一次。所以期望達到了。
如何緩解?
有兩種方法可以彌補 testify/suite 和測試之間的差距。
第一種方法是在每個測試方法之前建立一個新的模擬,如下所示。
func (suite *ExampleTestSuite) TestExample() { suite.Equal(5, suite.VariableThatShouldStartAtFive) }
有時,由於多種原因,它在您的專案中並不實用,例如為每個測試案例設定一個伺服器實例太昂貴。然後你可以嘗試另一個方向,即每次測試後手動斷言。
第二個是在每個測試方法的最後加上對 AssertExpectations 的呼叫。例如,在 TearDownTest 中呼叫 AssertExpectations,它在每個測試方法之後執行。
func TestXxx(*testing.T)
以上是揭露 Go 中隱藏的測試陷阱:避免誤報的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

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

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

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

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

熱門文章

熱工具

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

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

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

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

Go語言在構建高效且可擴展的系統中表現出色,其優勢包括:1.高性能:編譯成機器碼,運行速度快;2.並發編程:通過goroutines和channels簡化多任務處理;3.簡潔性:語法簡潔,降低學習和維護成本;4.跨平台:支持跨平台編譯,方便部署。

Golang在並發性上優於C ,而C 在原始速度上優於Golang。 1)Golang通過goroutine和channel實現高效並發,適合處理大量並發任務。 2)C 通過編譯器優化和標準庫,提供接近硬件的高性能,適合需要極致優化的應用。

Golang和Python各有优势:Golang适合高性能和并发编程,Python适用于数据科学和Web开发。Golang以其并发模型和高效性能著称,Python则以简洁语法和丰富库生态系统著称。

Golang在性能和可擴展性方面優於Python。 1)Golang的編譯型特性和高效並發模型使其在高並發場景下表現出色。 2)Python作為解釋型語言,執行速度較慢,但通過工具如Cython可優化性能。

Golang和C 在性能競賽中的表現各有優勢:1)Golang適合高並發和快速開發,2)C 提供更高性能和細粒度控制。選擇應基於項目需求和團隊技術棧。

C 更適合需要直接控制硬件資源和高性能優化的場景,而Golang更適合需要快速開發和高並發處理的場景。 1.C 的優勢在於其接近硬件的特性和高度的優化能力,適合遊戲開發等高性能需求。 2.Golang的優勢在於其簡潔的語法和天然的並發支持,適合高並發服務開發。

goimpactsdevelopmentpositationality throughspeed,效率和模擬性。 1)速度:gocompilesquicklyandrunseff,IdealforlargeProjects.2)效率:效率:ITScomprehenSevestAndardArdardArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdEcceSteral Depentencies,增強的Depleflovelmentimency.3)簡單性。

Golang和C 在性能上的差異主要體現在內存管理、編譯優化和運行時效率等方面。 1)Golang的垃圾回收機制方便但可能影響性能,2)C 的手動內存管理和編譯器優化在遞歸計算中表現更為高效。
