Go 関数の単体テストを行う場合は、次の落とし穴に注意してください。外部リソースへの依存を避け、スタブとモックを使用して依存関係を分離します。エラーをチェックし、無視しないでください。プライベート メソッドをテストするには、リフレクションまたは名前変更を使用します。同期プリミティブを使用して、同時実行時の競合状態を回避します。
Go 関数の単体テストの罠と考慮事項
単体テストは、コードの品質を確保するための重要な実践です。 Go では、テストには testing
パッケージを使用します。単体テストは比較的単純ですが、注意すべき落とし穴や注意点がいくつかあります。
1. 外部リソースへの依存
単体テストでは、外部リソース (データベースやネットワーク呼び出しなど) に依存せず、テスト対象のコードを分離する必要があります。これを行うには、スタブ、モック、またはテスト ダブルを使用して外部依存関係を分離します。
例 (スタブ):
type DatabaseClient interface { GetUser(id int) (*User, error) } // DbClientStub 是 DatabaseClient 的桩 type DbClientStub struct { GetResult *User GetError error } func (s *DbClientStub) GetUser(id int) (*User, error) { return s.GetResult, s.GetError }
2. エラーを無視する
テストではエラーを無視したくなることがあります。特に、通常のコードパスをテストする場合に当てはまります。ただし、これにより、デバッグが困難な問題が発生したり、未処理のエラーでコードが失敗したりする可能性があります。可能な場合は、エラーを常にチェックし、それに応じて処理する必要があります。
例:
func GetUser(id int) (*User, error) { // ... 从数据库中获取用户 // **不要忽略错误!** if err != nil { return nil, err } return user, nil }
3. プライベート メソッドのテスト
Go 言語のプライベート メソッド (小文字の名前) は通常、 used インターフェイス メソッドを実装するか、実装の詳細を非表示にします。ただし、外部から直接テストすることはできません。プライベート メソッドをテストするには、いくつかの方法があります。
reflect
パッケージを使用して、プライベート メソッドにアクセスします。 例 (反省):
func TestPrivateMethod(t *testing.T) { // 使用反射访问私有方法 value := reflect.ValueOf(myPackage.myPrivateMethod) result := value.Call([]reflect.Value{reflect.ValueOf(123)}) // 检查结果 if result[0].Int() != 456 { t.Errorf("Expected 456, got %d", result[0].Int()) } }
4. 競合状態
Go の同時実行性により競合が発生します。という条件が考えられます。単体テストでは、同時実行の Goroutine で同期プリミティブ (sync.Mutex など) を使用するなどして、競合状態を回避するように注意する必要があります。
例 (sync.Mutex
を使用):
var userMap sync.Map func TestConcurrentUserMap(t *testing.T) { // 创建 goroutine 并发访问 userMap for i := 0; i < 1000; i++ { go func(i int) { userMap.LoadOrStore(i, "User"+strconv.Itoa(i)) }(i) } // 等待所有 goroutine 完成 time.Sleep(time.Millisecond) // 验证 userMap 是否包含所有预期的键 for i := 0; i < 1000; i++ { if _, ok := userMap.Load(i); !ok { t.Errorf("userMap doesn't contain key %d", i) } } }
以上がGo 関数の単体テストの落とし穴と考慮事項の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。