>백엔드 개발 >Golang >Go 'for...range' 루프에 포인터를 추가하면 예기치 않은 결과가 생성되는 이유는 무엇입니까?

Go 'for...range' 루프에 포인터를 추가하면 예기치 않은 결과가 생성되는 이유는 무엇입니까?

Barbara Streisand
Barbara Streisand원래의
2024-12-30 17:03:12793검색

Why Does Appending Pointers in a Go `for...range` Loop Produce Unexpected Results?

Go에서 추가를 사용한 이상한 동작 이해

Go에서 추가 기능은 슬라이스에서 작동하여 새로운 요소를 추가합니다. 그러나 for range 루프 내의 배열 요소에 포인터를 추가할 때 사용자는 예기치 않은 동작을 경험할 수 있습니다.

다음 예를 고려하세요.

import "fmt"

type Foo struct {
    val int
}

func main() {
    var a = make([]*Foo, 1)
    a[0] = &Foo{0}

    var b = [3]Foo{Foo{1}, Foo{2}, Foo{3}}
    for _, e := range b {
        a = append(a, &e)
    }

    for _, e := range a {
        fmt.Printf("%v ", *e)
    }
}

예상 출력: {0} {1} {2} {3}

실제 출력: {0} {3} {3} {3}

동작 이유

이 예기치 않은 동작은 for range 루프가 원본 요소 자체가 아닌 요소의 복사본을 반복하기 때문에 발생합니다. 이 경우 루프 변수 e는 배열의 현재 요소 복사본을 보유합니다. 슬라이스가 추가되면 모든 반복에 대해 동일한 메모리 위치를 참조하는 루프 변수의 주소가 추가됩니다. 결과적으로 배열의 마지막 요소가 발견되면 추가된 모든 주소는 동일한 요소를 가리킵니다.

Fix

이 문제를 해결하려면 추가 기능을 다음과 같이 설정해야 합니다. 루프 변수가 아닌 원래 배열 요소의 주소와 함께 사용됩니다. 수정된 코드는 다음과 같습니다.

for i := range b {
    a = append(a, &b[i])
}

이 수정을 통해 예상한 대로 출력됩니다. {0} {1} {2} {3}.

결론

Go에서 포인터 유형과 포인터가 아닌 유형의 차이점을 이해하는 것은 메모리 참조를 적절하게 처리하는 데 중요합니다. for range 루프를 사용할 때 원본 요소에 액세스해야 하는지 아니면 복사본에 액세스해야 하는지 고려하고 이에 따라 적절한 구문을 사용하는 것이 중요합니다.

위 내용은 Go 'for...range' 루프에 포인터를 추가하면 예기치 않은 결과가 생성되는 이유는 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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