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

(시리즈 글이 2개 있습니다.)
.NET Framework: 939. C# - 전위/후위 증감 연산자에 대한 오버로딩 구현
; https://www.sysnet.pe.kr/2/0/12330

.NET Framework: 941. C# - 전위/후위 증감 연산자에 대한 오버로딩 구현 (2)
; https://www.sysnet.pe.kr/2/0/12333




C# - 전위/후위 증감 연산자에 대한 오버로딩 구현 (2)

아래의 내용에 이어,

C# - 전위/후위 증감 연산자에 대한 오버로딩 구현
; https://www.sysnet.pe.kr/2/0/12330

덧글에서 또다시 의문을 제기했는데,

그런데 아래 내용에서 another에 value+1의 값을 넣었으면 연산 순서에 따라 연산이 된 value+1의 값이 another라는 instance에 들어가야 하는데 그렇지 않고 왜 반환되는 값은 value인 것인지 이해가 가지 않습니다.

public static Integer operator ++(Integer instance)
{
    Integer another = new Integer(instance._value + 1);
    return another;
}


어찌 보면 신기할 수 있지만, 사실 일반적인 전위/후위 원칙과 별반 다르지 않습니다. 이미 기존에도 설명했지만, 동일한 ++의 코드임에도 전/후위 표기에 따라 C# 컴파일러는 Increment/Decrement 연산자의 사용 코드를 다음과 같이 풀어서 번역합니다.

int n = 5;
int value = ++ n;

==> ++를 사용하는 측에서 다음과 같이 코드 번역

int n = 5;
n = n + 1; // 값을 증가시키고,
int value = n; // 이후에 대입

int n = 5;
int value = n ++;

==> ++를 사용하는 측에서 다음과 같이 코드 번역

int n = 5;
int value = n; // 값을 먼저 대입하고,
n = n + 1; // 이후에 증가

그러니까, 재정의된 전위/후위 연산자도 내부 코드는 같지만 사용하는 측에서 다음과 같이 번역해 버리면 그만입니다.

{
    Integer n = new Integer(5);
    Integer value = n++;
}
==>
        {
            Integer n = new Integer(5);
            Integer temp = Integer.operator ++(n);
            Integer value = n;
            n = temp;
        }

{
    Integer n = new Integer(5);
    Integer value = ++n;
}
==>
        {
            Integer n = new Integer(5);
            Integer value = Integer.operator ++(n);
            n = value;
        }

실제로 저렇게 번역이 되는지 확인하고 싶다면 IL 코드를 보면 됩니다.

.locals init (
	[0] class Integer n,
	[1] class Integer 'value',
	[2] class Integer n,
	[3] class Integer 'value'
)

/* 0x0000025C 00           */ IL_0000: nop
/* 0x0000025D 00           */ IL_0001: nop
/* 0x0000025E 1B           */ IL_0002: ldc.i4.5
/* 0x0000025F 7303000006   */ IL_0003: newobj    instance void Integer::.ctor(int32)
/* 0x00000264 0A           */ IL_0008: stloc.0
/* 0x00000265 06           */ IL_0009: ldloc.0
/* 0x00000266 25           */ IL_000A: dup
/* 0x00000267 2804000006   */ IL_000B: call      class Integer Integer::op_Increment(class Integer)
/* 0x0000026C 0A           */ IL_0010: stloc.0
/* 0x0000026D 0B           */ IL_0011: stloc.1
/* 0x0000026E 07           */ IL_0012: ldloc.1
/* 0x0000026F 280F00000A   */ IL_0013: call      void [mscorlib]System.Console::WriteLine(object)
/* 0x00000274 00           */ IL_0018: nop
/* 0x00000275 00           */ IL_0019: nop
/* 0x00000276 00           */ IL_001A: nop
/* 0x00000277 1B           */ IL_001B: ldc.i4.5
/* 0x00000278 7303000006   */ IL_001C: newobj    instance void Integer::.ctor(int32)
/* 0x0000027D 0C           */ IL_0021: stloc.2
/* 0x0000027E 08           */ IL_0022: ldloc.2
/* 0x0000027F 2804000006   */ IL_0023: call      class Integer Integer::op_Increment(class Integer)
/* 0x00000284 25           */ IL_0028: dup
/* 0x00000285 0C           */ IL_0029: stloc.2
/* 0x00000286 0D           */ IL_002A: stloc.3
/* 0x00000287 09           */ IL_002B: ldloc.3
/* 0x00000288 280F00000A   */ IL_002C: call      void [mscorlib]System.Console::WriteLine(object)
/* 0x0000028D 00           */ IL_0031: nop
/* 0x0000028E 00           */ IL_0032: nop
/* 0x0000028F 2A           */ IL_0033: ret




답변을 하다 보니, 재미있는 점이 눈에 띕니다. 전위 연산자의 경우에는 상관없지만, 후위 연산자의 경우에는, (후위 연산자가 꽤나 문제군요. ^^)

{
    Integer n = new Integer(5);
    Integer value = ++n;
}
==>
        {
            Integer n = new Integer(5);
            Integer value = Integer.operator ++(n);
            n = value;
        }

결국 같은 인스턴스가 n과 value에 들어가 참조 값이 같아집니다. 실제로 이를 다음의 코드로 테스트해볼 수 있습니다.

{
    Integer n = new Integer(5);
    Integer value = ++n;

    value.Increment(); //
    value.Increment(); // value의 값을 변경했지만,
    value.Increment(); //
    Console.WriteLine(value); // 출력 결과: 9
    Console.WriteLine(n); // 출력 결과: 9 (n의 값도 함께 변경)
}

public class Integer
{
    int _value;

    // ...[생략]...

    internal void Increment()
    {
        _value++;
    }
}

이것은 일종의 side-effect 일 수 있는데, 이런 부분을 고려한다면 연산자 오버로딩을 포함한 타입은 가능한 class보다는 struct로 구현하는 것이 권장됩니다.




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







[최초 등록일: ]
[최종 수정일: 12/18/2020]

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

비밀번호

댓글 작성자
 




1  2  3  4  5  6  7  8  [9]  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13718정성태8/27/20247434오류 유형: 921. Visual C++ - error C1083: Cannot open include file: 'float.h': No such file or directory [2]
13717정성태8/26/20247028VS.NET IDE: 192. Visual Studio 2022 - Windows XP / 2003용 C/C++ 프로젝트 빌드
13716정성태8/21/20246767C/C++: 167. Visual C++ - 윈도우 환경에서 _execv 동작 [1]
13715정성태8/19/20247383Linux: 78. 리눅스 C/C++ - 특정 버전의 glibc 빌드 (docker-glibc-builder)
13714정성태8/19/20246761닷넷: 2295. C# 12 - 기본 생성자(Primary constructors) (책 오타 수정) [3]
13713정성태8/16/20247488개발 환경 구성: 721. WSL 2에서의 Hyper-V Socket 연동
13712정성태8/14/20247225개발 환경 구성: 720. Synology NAS - docker 원격 제어를 위한 TCP 바인딩 추가
13711정성태8/13/20248074Linux: 77. C# / Linux - zombie process (defunct process) [1]파일 다운로드1
13710정성태8/8/20248000닷넷: 2294. C# 13 - (6) iterator 또는 비동기 메서드에서 ref와 unsafe 사용을 부분적으로 허용파일 다운로드1
13709정성태8/7/20247766닷넷: 2293. C# - safe/unsafe 문맥에 대한 C# 13의 (하위 호환을 깨는) 변화파일 다운로드1
13708정성태8/7/20247550개발 환경 구성: 719. ffmpeg / YoutubeExplode - mp4 동영상 파일로부터 Audio 파일 추출
13707정성태8/6/20247791닷넷: 2292. C# - 자식 프로세스의 출력이 4,096보다 많은 경우 Process.WaitForExit 호출 시 hang 현상파일 다운로드1
13706정성태8/5/20247899개발 환경 구성: 718. Hyper-V - 리눅스 VM에 새로운 디스크 추가
13705정성태8/4/20248167닷넷: 2291. C# 13 - (5) params 인자 타입으로 컬렉션 허용 [2]파일 다운로드1
13704정성태8/2/20248126닷넷: 2290. C# - 간이 dotnet-dump 프로그램 만들기파일 다운로드1
13703정성태8/1/20247450닷넷: 2289. "dotnet-dump ps" 명령어가 닷넷 프로세스를 찾는 방법
13702정성태7/31/20247861닷넷: 2288. Collection 식을 지원하는 사용자 정의 타입을 CollectionBuilder 특성으로 성능 보완파일 다운로드1
13701정성태7/30/20248130닷넷: 2287. C# 13 - (4) Indexer를 이용한 개체 초기화 구문에서 System.Index 연산자 허용파일 다운로드1
13700정성태7/29/20247752디버깅 기술: 200. DLL Export/Import의 Hint 의미
13699정성태7/27/20248251닷넷: 2286. C# 13 - (3) Monitor를 대체할 Lock 타입파일 다운로드1
13698정성태7/27/20248209닷넷: 2285. C# - async 메서드에서의 System.Threading.Lock 잠금 처리파일 다운로드1
13697정성태7/26/20247932닷넷: 2284. C# - async 메서드에서의 lock/Monitor.Enter/Exit 잠금 처리파일 다운로드1
13696정성태7/26/20247469오류 유형: 920. dotnet publish - error NETSDK1047: Assets file '...\obj\project.assets.json' doesn't have a target for '...'
13695정성태7/25/20247455닷넷: 2283. C# - Lock / Wait 상태에서도 STA COM 메서드 호출 처리파일 다운로드1
13694정성태7/25/20247920닷넷: 2282. C# - ASP.NET Core Web App의 Request 용량 상한값 (Kestrel, IIS)
13693정성태7/24/20247244개발 환경 구성: 717. Visual Studio - C# 프로젝트에서 레지스트리에 등록하지 않은 COM 개체 참조 및 사용 방법파일 다운로드1
1  2  3  4  5  6  7  8  [9]  10  11  12  13  14  15  ...