>  기사  >  백엔드 개발  >  ## 왜 Go에서 값에 대한 포인터 메서드를 호출할 수 있나요?

## 왜 Go에서 값에 대한 포인터 메서드를 호출할 수 있나요?

Mary-Kate Olsen
Mary-Kate Olsen원래의
2024-10-25 07:55:02770검색

##  Why Can I Call a Pointer Method on a Value in Go?

Go에서 구문 혼란을 호출하는 수신자 메서드

리시버에 대한 포인터와 값의 차이점을 탐색할 때 값 메서드는 다음과 같을 수 있다는 점에 유의하세요. 포인터와 값 모두에서 호출되는 반면 포인터 메서드는 포인터로만 제한됩니다. 이러한 제한은 수신자를 수정하는 포인터 메서드의 기능으로 인해 발생하며, 값의 복사본에서 호출되면 수정 사항이 삭제됩니다.

그러나 값에 대해 호출되는 포인터 메서드를 보여주는 코드 예제는 혼란을 야기합니다. 코드는 다음과 같습니다.

<code class="go">package main

import (
    "fmt"
    "reflect"
)

type age int

func (a age) String() string {
    return fmt.Sprintf("%d year(s) old", int(a))
}

func (a *age) Set(newAge int) {
    if newAge >= 0 {
        *a = age(newAge)
    }
}

func main() {
    var vAge age = 5
    pAge := new(age)

    fmt.Printf("Type information:\n\tvAge: %v\n\tpAge: %v\n", reflect.TypeOf(vAge),
        reflect.TypeOf(pAge))

    fmt.Printf("vAge.String(): %v\n", vAge.String())
    fmt.Printf("vAge.Set(10)\n")
    vAge.Set(10)
    fmt.Printf("vAge.String(): %v\n", vAge.String())

    fmt.Printf("pAge.String(): %v\n", pAge.String())
    fmt.Printf("pAge.Set(10)\n")
    pAge.Set(10)
    fmt.Printf("pAge.String(): %v\n", pAge.String())
}</code>

문서에 불가능하다고 명시되어 있음에도 불구하고 이 코드는 오류 없이 컴파일되고 실행됩니다. 그렇다면 이러한 명백한 불일치의 원인은 무엇입니까?

답은 주소 지정 가능한 값의 개념에 있습니다. Go 사양에서는 "x의 메소드 세트(유형)에 m이 포함되고 인수 목록이 m의 매개변수 목록에 할당될 수 있는 경우 메소드 호출 x.m()이 유효합니다."라고 정의합니다. 또한 "x가 주소 지정 가능하고 &x의 메서드 세트에 m이 포함된 경우 x.m()은 (&x).m()의 약어입니다."

이 경우 vAge는 주소 지정이 가능합니다. 즉 유효한 메모리 주소가 있음을 의미합니다. . vAge.Set()을 호출할 때 컴파일러는 (&vAge).Set().

과 동일한 값의 주소를 사용하여 포인터 수신기 메서드 Set()을 호출할 수도 있음을 인식합니다.

위 내용은 ## 왜 Go에서 값에 대한 포인터 메서드를 호출할 수 있나요?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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