>  기사  >  백엔드 개발  >  Go\의 `fmt.Stringer`에서 `t`와 `*t`를 수신자로 사용하면 데드 루프가 발생하는 이유는 무엇입니까?

Go\의 `fmt.Stringer`에서 `t`와 `*t`를 수신자로 사용하면 데드 루프가 발생하는 이유는 무엇입니까?

Patricia Arquette
Patricia Arquette원래의
2024-11-22 22:49:12665검색

Why does using `t` vs `*t` as a receiver in Go's `fmt.Stringer` lead to a dead loop?

t와 *t의 차이점 이해: fmt 패키지의 사례 연구

Go에서 fmt 패키지는 강력한 형식 지정 기능을 제공합니다. , 개발자가 값이 인쇄되는 방식을 사용자 정의할 수 있습니다. 그러나 t와 *t 사이의 미묘한 차이로 인해 예상치 못한 동작, 즉 수신자 메서드 내에서 사용될 때 데드 루프가 발생할 수 있습니다.

다음 코드 조각을 고려하세요.

package main

import "fmt"

type TT struct {
    a int
    b float32
    c string
}

func (t *TT) String() string {
    return fmt.Sprintf("%+v", *t)
}

func main() {
    tt := &TT{3, 4, "5"}
    fmt.Printf(tt.String())
}

이 코드 성공적으로 실행되어 'tt' 변수의 내용을 "tt={a:3, b:4, c:5}"로 인쇄합니다. 그러나 String 메서드를 다음과 같이 수정하면

func (t *TT) String() string {
    return fmt.Sprintf("%+v", t)
}

데드 루프가 발생합니다. 그 이유는 fmt 패키지가 String() 메서드(fmt.Stringer 인터페이스 구현)를 구현하는 값을 처리하는 방식에 있습니다.

'tt' 값이 fmt.Println에 전달되면 fmt 패키지는 TT 유형이 String() 메소드를 구현하는지 확인합니다. 포인터 수신기(*TT)를 사용하여 String() 메서드를 정의했으므로 TT에 설정된 메서드에는 String()이 포함되지 않습니다. 결과적으로 fmt 패키지는 tt.String()을 호출하지 않습니다.

반면, 수정된 코드 조각에서는 *TT의 수신자 유형으로 String() 메서드를 정의합니다. 이 경우 TT에 설정된 메소드에는 String()이 포함됩니다. 그러나 *TT 유형의 tt를 전달하므로 tt.String()을 호출하면 본질적으로 String() 메서드가 재귀적으로 호출되어 무한 루프가 발생합니다.

이 문제를 방지하려면 다음을 수행하는 것이 좋습니다. String() 메소드에 대해 다른 수신자 유형을 사용하십시오. 한 가지 접근 방식은 type 키워드를 사용하여 새 유형을 만들고 전달되는 값에 대해 유형 변환을 수행하는 것입니다. 이렇게 하면 메소드가 없는 새 유형이 생성되어 무한 루프를 피할 수 있습니다.

예:

func (t TT) String() string {
    type TT2 TT
    return fmt.Sprintf("%+v", TT2(t))
}

이 경우 t를 TT2로 변환하면 String() 메소드가 TT2에는 String() 메소드가 없으므로 재귀적으로 호출됩니다.

다음의 맥락에서 t와 *t의 차이점 이해 fmt 패키지는 예상치 못한 동작을 방지하고 효율적인 코드 실행을 보장하는 데 매우 중요합니다.

위 내용은 Go\의 `fmt.Stringer`에서 `t`와 `*t`를 수신자로 사용하면 데드 루프가 발생하는 이유는 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.