>  기사  >  백엔드 개발  >  리플렉션을 사용하여 구조체 필드 값을 설정할 때 SetCan()이 항상 False를 반환하는 문제를 해결하는 방법은 무엇입니까?

리플렉션을 사용하여 구조체 필드 값을 설정할 때 SetCan()이 항상 False를 반환하는 문제를 해결하는 방법은 무엇입니까?

Barbara Streisand
Barbara Streisand원래의
2024-10-24 04:19:30483검색

How to Overcome SetCan() Always Returning False When Setting Struct Field Values Using Reflection?

구조체에 대해 SetString을 사용하여 Reflection 탐색

Reflection은 Go 구조를 동적으로 조작하기 위한 강력한 도구를 제공합니다. 이 예에서는 리플렉션을 사용하여 구조체 필드의 값을 설정하려고 할 때 일반적인 문제에 직면합니다. CanSet()은 항상 false를 반환합니다. 이러한 방해로 인해 현장 수정이 방지되어 우리는 어려움에 빠지게 됩니다.

위험 식별

제공된 코드 조각은 두 가지 근본적인 오류를 강조합니다.

  1. 포인터 대신 값 전달: 값 기반 구조체는 리플렉션을 통해 수정할 수 없습니다. 대신 구조체에 대한 포인터를 전달하여 모든 변경 사항이 실제 객체에 적용되도록 해야 합니다.
  2. 잘못된 요소 타겟팅: 코드는 처음에는 ProductionInfo 구조체가 아닌 전체 ProductionInfo 구조체에서 작동합니다. Field1 값을 변경해야 하는 특정 항목입니다. 이를 수정하려면 원하는 Entry 요소에 집중해야 합니다.

문제 해결

이러한 함정을 해결한 후 코드를 개선할 수 있습니다.

<code class="go">func SetField(source interface{}, fieldName string, fieldValue string) {
    v := reflect.ValueOf(source).Elem() // Obtain the value of the pointed object
    fmt.Println(v.FieldByName(fieldName).CanSet())
    if v.FieldByName(fieldName).CanSet() {
        v.FieldByName(fieldName).SetString(fieldValue)
    }
}</code>

수정된 SetField() 함수에서 다음을 수행합니다.

  1. Elem()을 사용하여 가리키는 개체의 값으로 이동하여 실제 구조체에서 작동하는지 확인합니다.
  2. FieldByName()을 사용하여 수정하려는 특정 필드에 액세스합니다.

코드 실행

이러한 수정을 통해 이제 코드는 Field1 값을 성공적으로 업데이트했습니다.

<code class="go">func main() {
    source := ProductionInfo{}
    source.StructA = append(source.StructA, Entry{Field1: "A", Field2: 2})

    fmt.Println("Before: ", source.StructA[0])
    SetField(&source.StructA[0], "Field1", "NEW_VALUE")
    fmt.Println("After: ", source.StructA[0])
}</code>

출력:

Before:  {A 2}
true
After:  {NEW_VALUE 2}

결과는 Entry 구조 내에서 Field1이 성공적으로 수정되었음을 보여줍니다.

위 내용은 리플렉션을 사용하여 구조체 필드 값을 설정할 때 SetCan()이 항상 False를 반환하는 문제를 해결하는 방법은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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