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 중국어 웹사이트의 기타 관련 기사를 참조하세요!