Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

Golang - 구조체의 slice 필드를 Reflection을 이용해 변경하는 방법

우선, 입문부터 해볼까요? ^^ 간단하게 slice를 만들고 이것에 요소를 추가하는 경우 이런 식으로 코딩할 수 있습니다.

buf := make([]string, 0)
buf := append(buf, "test")

위에서 "append"하는 과정을 Reflection으로 구현한다면 reflect를 이용해 유사한 방식으로 처리하게 됩니다.

value := reflect.ValueOf(&buf)
value := reflect.Append(value.Elem(), reflect.ValueOf("test"))

// 만약 포인터로 전달하지 않으면 reflect.Append 호출에서 예외 발생 - reflect: call of reflect.Value.Elem on slice Value
// value := reflect.ValueOf(buf)

보는 바와 같이, slice가 append를 이용할 때도 그 반환값을 다시 buf 변수에 할당해야만 했던 것처럼, reflect를 이용하는 것도 그와 유사한 것입니다. 바로 이런 점 때문에 결국 buf 변수와 reflect.Append가 반환한 value 변수는 같은 상태를 가리키지 않습니다. 즉, 위와 같은 처리가 원본 변수인 buf에 영향을 주는 것은 아닙니다.

fmt.Printf("%v, %v", len(buf), value)
/* 출력 결과
0, [test]
*/

따라서, reflection으로 처리한 결과를 다시 원본 변수인 buf에 대입하려면 다음과 같은 식의 코딩을 통해 reflect.Value로부터 값을 변환해 줘야 합니다.

buf = value.Interface().([]string)
fmt.Printf("%v, %v\n", len(buf), buf[0]) 
/* 출력 결과
0, test
*/




자, 그럼 타입의 멤버로 slice가 있다면 어떻게 될까요?

type MyType struct {
    Buf []string
}

기본적인 코드는 다음과 같이 시작할 수 있습니다.

t := MyType{}
t.Buf = make([]string, 0)

tValue := reflect.ValueOf(t)

bufValue := tValue.FieldByName("Buf") // 필드 이름으로 조회

bufType := bufValue.Type()
fmt.Printf("%v, %v\n", bufType, bufValue)

fmt.Printf("%v\n", bufType.Elem().Kind())

/* 출력 결과
[]string, []
string
*/

그럼 마찬가지로 slice에 멤버를 추가하는 것도 동일하게 반영할 수 있고,

bufValue = reflect.Append(bufValue, reflect.ValueOf("test"))
fmt.Printf("%v, %v", len(t.Buf), bufValue)
/* 출력 결과
0, [test]
*/

원본에 적용하는 것도 상황이 허락된다면 이렇게 할 수 있습니다.

bufValue = reflect.Append(bufValue, reflect.ValueOf("test"))
t.Buf = bufValue.Interface().([]string)
fmt.Printf("%v, %v", len(t.Buf), t.Buf[0])
/* 출력 결과
1, test
*/

그런데, 여기서 문제가 있습니다. 대개의 경우 reflection을 사용할 때는 대상 인스턴스를 interface {}로 받게 될 것입니다.

t := MyType{}
t.buf = make([]string, 0)

Modify(t)

func Modify(inst interface {}) {
     // ... reflection ...
}

위와 같은 상황에서, slice가 아닌 다른 타입이었다면 reflect.Value의 SetString, SetInt 등을 이용해 값을 설정하는 것이 가능합니다. 그런데, slice 유형이라면 단순히 Set 함수를 이용하는 경우 예외가 발생합니다.

changed := reflect.Append(bufValue, reflect.ValueOf("test"))
bufValue.Set(changed) // 예외 발생:  reflect: reflect.Value.Set using unaddressable value

다행히 예외 메시지에 답이 있는데요, 애당초 구조체 인스턴스를 포인터로 전달했어야 하고, reflection을 하려는 측에서도 Pointer인 경우 reflect.ValueOf/TypeOf를 했던 대상의 원본에 대해 한 번 더 Elem()을 호출해 대상 인스턴스를 가져와야 합니다. 아래의 코드는 그 상황을 보여줍니다.

func main() {
    t := MyType{}
    t.Buf = make([]string, 0)

    Modify(&t);

    /* 또는,
    t := &MyType{}
    t.Buf = make([]string, 0)

    Modify(t);
    */
}

func Modify(t interface{}) {

    tType := reflect.TypeOf(t)
    tValue := reflect.ValueOf(t)

    if tType.Kind() == reflect.Ptr {
        tType = tType.Elem()
        tValue = tValue.Elem()
    }

    bufValue := tValue.FieldByName("Buf")

    bufType := bufValue.Type()
    changed := reflect.Append(bufValue, reflect.ValueOf("test"))

    bufValue.Set(changed)
}




만약 타입의 멤버 이름을 소문자로 바꿔 접근성을 변경하면,

type MyType struct {
    buf []string
}

reflect.Append 코드에서 예외가 발생합니다.

// 예외 발생: reflect: reflect.Copy using value obtained using unexported field
bufValue = reflect.Append(bufValue, reflect.ValueOf("test"))

다른 언어와는 달리, reflection으로도 private 필드에 대한 접근은 할 수 없는 것입니다.




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]







[최초 등록일: ]
[최종 수정일: 8/23/2022]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 76  77  78  79  80  [81]  82  83  84  85  86  87  88  89  90  ...
NoWriterDateCnt.TitleFile(s)
11971정성태7/3/201921889개발 환경 구성: 447. Visual Studio Code에서 OpenCvSharp 개발 환경 구성
11970정성태7/2/201921808오류 유형: 552. 웹 브라우저에서 파일 다운로드 후 "Running security scan"이 끝나지 않는 문제
11969정성태7/2/201922385Math: 63. C# - 3층 구조의 신경망파일 다운로드1
11968정성태7/1/201929013오류 유형: 551. Visual Studio Code에서 Remote-SSH 연결 시 "Opening Remote..." 단계에서 진행되지 않는 문제 [1]
11967정성태7/1/201923193개발 환경 구성: 446. Synology NAS를 Windows 10에서 iSCSI로 연결하는 방법
11966정성태6/30/201921738Math: 62. 활성화 함수에 따른 뉴런의 출력을 그리드 맵으로 시각화파일 다운로드1
11965정성태6/30/201922253.NET Framework: 846. C# - 2차원 배열을 1차원 배열로 나열하는 확장 메서드파일 다운로드1
11964정성태6/30/201923052Linux: 20. C# - Linux에서의 Named Pipe를 이용한 통신
11963정성태6/29/201922591Linux: 19. C# - .NET Core Unix Domain Socket 사용 예제
11962정성태6/27/201920248Math: 61. C# - 로지스틱 회귀를 이용한 선형분리 불가능 문제의 분류파일 다운로드1
11961정성태6/27/201921153Graphics: 37. C# - PLplot - 출력 모음(Family File Output)
11960정성태6/27/201922095Graphics: 36. C# - PLplot의 16색 이상을 표현하는 방법과 subpage를 이용한 그리드 맵 표현
11959정성태6/27/201923380Graphics: 35. matplotlib와 PLplot의 한글 처리
11958정성태6/25/201927565Linux: 18. C# - .NET Core Console로 리눅스 daemon 프로그램 만드는 방법 [6]
11957정성태6/24/201924594Windows: 160. WMI 쿼리를 명령행에서 간단하게 수행하는 wmic.exe [2]
11956정성태6/24/201924449Linux: 17. CentOS 7에서 .NET Core Web App 실행 환경 구성 [1]
11955정성태6/20/201922513Math: 60. C# - 로지스틱 회귀를 이용한 분류파일 다운로드1
11954정성태6/20/201920410오류 유형: 550. scp - sudo: no tty present and no askpass program specified
11953정성태6/20/201918087오류 유형: 549. The library 'libhostpolicy.so' required to execute the application was not found in '...'
11952정성태6/20/201919396Linux: 16. 우분투, Centos의 Netbios 호스트 이름 풀이 방법
11951정성태6/20/201922654오류 유형: 548. scp 연결 시 "Permission denied" 오류 및 "WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED!" 경고
11950정성태6/18/201923799.NET Framework: 845. C# - 윈도우 작업 관리자와 리소스 모니터의 메모리 값을 구하는 방법
11949정성태6/18/201919146오류 유형: 547. CoreCLR Profiler 예제 프로젝트 빌드 시 컴파일 오류 유형
11948정성태6/17/201920487Linux: 15. 리눅스 환경의 Visual Studio Code에서 TFS 서버 연동
11947정성태6/17/201923401Linux: 14. 리눅스 환경에서 TFS 서버 연동
11946정성태6/17/201924558개발 환경 구성: 445. C# - MathNet으로 정규 분포를 따르는 데이터를 생성, PLplot으로 Histogram 표현파일 다운로드1
... 76  77  78  79  80  [81]  82  83  84  85  86  87  88  89  90  ...