Effective Go의 "포인터 대 값" 섹션에는 "값 메서드가 포인터와 값에 대해 호출될 수 있음"이 명시되어 있습니다. , 그러나 포인터 메서드는 포인터에서만 호출할 수 있습니다." 이는 포인터 메서드가 수신자를 수정할 수 있고 값 복사본에서 이를 호출하면 이러한 수정 사항이 삭제되기 때문입니다.
그러나 일부 개발자는 다음과 같은 상황에 직면했습니다. 문서화된 구문 규칙과 모순되는 것처럼 보이는 값에 대한 포인터 메서드를 호출합니다. 이러한 혼란을 명확히 하기 위해 이 동작을 더 자세히 살펴보겠습니다.
다음 Go 코드를 고려하세요.
<code class="go">package main import ( "fmt" "reflect" ) type age int func (a age) String() string { return fmt.Sprintf("%d yeasr(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("TypeOf =>\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>
이 예에서는 두 가지 메소드로 연령 유형을 정의합니다. )(값 메서드) 및 Set()(포인터 메서드). 코드는 두 가지 변수, 즉 age 유형의 값인 vAge와 age 값에 대한 포인터인 pAge를 생성합니다.
vAge.Set()가 유효한 구문이 아니어야 한다는 문서에도 불구하고 코드는 다음과 같이 컴파일됩니다. 오류. 이는 vAge가 주소 지정이 가능하기 때문입니다. 즉, 참조할 수 있는 메모리 주소가 있다는 의미입니다. 언어 사양에 따라 주소 지정 가능한 변수 x에 대한 메서드 호출 x.m()은 x의 메서드 집합에 m이 포함된 경우 유효합니다.
이 경우 vAge는 주소 지정이 가능하며 *age의 메서드 집합에 Set이 포함되어 있으므로 ()에서 vAge.Set() 호출은 (&vAge).Set()에 대한 약칭 표기법입니다. 기본적으로 & 연산자는 vAge에 대한 포인터를 얻기 위해 암시적으로 사용됩니다.
따라서 코드는 주소 지정 가능한 값에 대해 값 메서드를 호출하는 것이 해당 포인터에 대한 메서드를 호출하는 것과 동일하기 때문에 실제로 포인터에서 값 메서드를 호출할 수 있음을 보여줍니다. 값. 포인터 메서드 Set()은 여전히 포인터에서만 호출할 수 있지만 값에 주소를 지정할 수 있는 경우 암시적 역참조를 통해 vAge.Set() 구문이 유효합니다.
위 내용은 ## 왜 Go에서 값에 대한 포인터 메서드를 호출할 수 있나요?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!