Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 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# - 전위/후위 증감 연산자에 대한 오버로딩 구현

질문이 있어서, ^^

후위 증감 연산자 오버로딩 방법 좀 알려주세요
; https://www.sysnet.pe.kr/3/0/5389

답변을 정리해 봅니다.

우선, C#은 Increment / Decrement 연산자(Operator)에 대해 Overloading을 지원합니다.

Overloadable operators
; https://learn.microsoft.com/en-us/dotnet/csharp/language-reference/operators/operator-overloading#overloadable-operators

구현하기 전에, 증감 연산자에 대한 Prefix/Postfix 표현의 구현 방식을 비교해 설명할 필요가 있습니다. (어차피 증감 연산자의 구현 방식은 동일하므로, 이 글에서는 ++ 연산자에 대한 예제만 다루겠습니다.)

우선, Prefix-Increment의 경우 내부적인 구현은 다음과 같이 이뤄집니다.

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

==> 

int n = 5;
n = n + 1; // 값을 증가시키고,
int value = n; // 이후에 대입 (결과적으로 증가된 값으로 대입)

반면 Postfix-Increment의 경우는 살짝 다릅니다.

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

==>

int n = 5;
int value = n; // 값을 먼저 대입하고,
n = n + 1; // 이후에 증가 (결과적으로 증가되지 않은 값을 대입)

자, 그럼 여기 질문자가 (아마도) 구현했을 잘못된 오버로딩 예제를 보겠습니다.

public class Integer
{
    int _value;

    public Integer(int value)
    {
        _value = value;
    }

    public static Integer operator ++(Integer instance)
    {
        instance._value++;
        return instance;
    }

    public override string ToString()
    {
        return _value.ToString();
    }
}

위와 같이 해서, 전/후위 대입을 해보면,

{
    Integer n = new Integer(5);
    Integer value = n++;
    Console.WriteLine(value); // 출력 결과: 6
}

{
    Integer n = new Integer(5);
    Integer value = ++n;
    Console.WriteLine(value); // 출력 결과: 6
}

차이점이 없습니다. 혹시 원인을 발견하셨나요?




이 문제를 수정하려면, 2가지 방식이 있습니다. 첫 번째는, 해당 타입을 struct로 바꾸면 됩니다.

public struct Integer
{
    // ...[생략]...
}

값 형식으로 바뀌었기 때문에, "n ++"과 "++ n"에 대한 연산에서,

// struct Integer인 경우

{   // Integer value = n ++;

    Integer value = n; // struct이므로 값 복사 발생
    n = n + 1;         // value의 값에 영향을 주지 않음
}

{   // Integer value = ++ n;

    n = n + 1;          // n 자체의 값이 증가 후,
    value = n;          // 값 복사가 되었으므로 value 값에 반영 
}

각각 풀어진 코드를 보면 정상적으로 동작하게 되는 이유를 알 수 있습니다. 위의 코드를 보면, class로 구현했을 때는 왜 비정상적인 동작을 했는지도,

// class Integer인 경우

{   // Integer value = n ++;

    Integer value = n; // class이므로 참조 복사 발생
    n = n + 1;         // n의 값이 바뀌지만 참조를 유지하는 value에도 영향
}

{   // Integer value = ++ n;

    n = n + 1;          // n의 값이 증가 후,
    value = n;          // value로 참조 복사
}

이해가 됩니다. 그렇다면, struct로 바꾸는 것이 여의치 않는 타입이라면 어떻게 해야 할까요? 그런 경우라면, ++ 연산자를 재정의하는 코드를 다음과 같이 바꿔야 합니다.

public class Integer
{
    // ...[생략]...

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

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

보는 바와 같이 새로운 인스턴스를 생성해 반환하는 식이므로, 이제 전/후의 표기에 따른 영향이 반영됩니다.

{   // Integer value = n ++;

    Integer value = n; // n의 복사본에 대한 참조가 대입되었고,
    n = n + 1;         // n의 값은 변경되지만 value에는 영향을 주지 않음
}

{   // Integer value = ++ n;

    n = n + 1;         // n의 값이 증가 후,
    value = n;         // value에 대입되므로 증가된 값 반영
}




정리해 보면, 결국 이것도 "값 형식"과 "참조 형식"에 대한 이해를 필요로 하는 문제였던 것입니다. 그리고, 웹상에 있는 C# 자료들이 대체로 연산자 재정의를 할 때 대상 인스턴스의 값을 직접 바꾸지 않고 (불변immutable 타입 다루듯이) 또 다른 인스턴스를 생성해 변경하는 식으로 구현하는 이유가 바로 value/ref 타입의 차이점에 기인합니다.

어쨌든, 이 글에서는 2가지 방식으로 구현을 했지만 struct/class의 차이점에 영향을 받지 않는 두 번째 방법을 더 권장합니다.

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 3/9/2024]

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

비밀번호

댓글 작성자
 



2020-09-20 04시21분
[질문자] 운영자님 답변해주셔서 감사합니다.
원 질문글의 작성자입니다.
아직 의문이 제대로 해결되지않습니다.

아래의 내용인 참조 복사를 가져간 instance는 참조 값이 변화되면 참조를 복사한 instance에도 영향을 준다는 내용은 이해하였습니다.

// class Integer인 경우
{ // Integer value = n ++;

    Integer value = n; // class이므로 참조 복사 발생
    n = n + 1; // n의 값이 바뀌지만 참조를 유지하는 value에도 영향
}

{ // Integer value = ++ n;

    n = n + 1; // n의 값이 증가 후,
    value = n; // value로 참조 복사
}

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

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

또한 위의 내용 하나만 구현하면 전위/후위 연산자가 모두 구현되는 것도 이해가 가지않습니다.
[guest]
2020-09-20 08시51분
C# - 전위/후위 증감 연산자에 대한 오버로딩 구현 (2)
; https://www.sysnet.pe.kr/2/0/12333
정성태

... 91  92  93  94  95  96  97  98  99  100  [101]  102  103  104  105  ...
NoWriterDateCnt.TitleFile(s)
11404정성태12/16/201729833.NET Framework: 709. C# - OpenCvSharp을 이용한 동영상(avi, mp4, ...) 처리 + Direct2D [7]파일 다운로드1
11403정성태12/16/201732454.NET Framework: 708. C# - OpenCvSharp을 이용한 동영상(avi, mp4, ...) 처리 [3]파일 다운로드1
11402정성태12/15/201737155.NET Framework: 707. OpenCV 응용 프로그램을 C#으로 구현 - OpenCvSharp [2]파일 다운로드1
11401정성태12/15/201726096.NET Framework: 706. C# - SharpDX + DXGI를 이용한 윈도우 화면 캡처 소스 코드 + Direct2D 출력 [2]파일 다운로드1
11400정성태12/14/201728964.NET Framework: 705. C# - SharpDX + DXGI를 이용한 윈도우 화면 캡처 소스 코드 [9]파일 다운로드1
11399정성태12/13/201717496.NET Framework: 704. Win32 API의 UnionRect를 닷넷 BCL의 Rectangle.Union으로 바꿀 때 주의 사항
11398정성태12/13/201717699오류 유형: 442. ASP.NET Core Web Application (on .NET Framework) 프로젝트에서 외부 라이브러리 동적 로드 시 런타임 버전 문제파일 다운로드1
11397정성태12/12/201720269.NET Framework: 703. 양자 컴퓨팅을 위한 마이크로소프트의 Q# 언어
11396정성태12/8/201742639개발 환경 구성: 343. Visual Studio - 리눅스 용 프로젝트의 인텔리센스를 위한 헤더 파일 처리 방법 [3]
11395정성태12/8/201718552오류 유형: 441. 이벤트 로그 - Time Provider NtpClient: No valid response has been received from domain controller
11394정성태12/8/201718170개발 환경 구성: 342. 비주얼 스튜디오에서 실행하던 ASP.NET Core (.NET Framework) 응용 프로그램을 명령행에서 실행하는 방법
11393정성태12/7/201722697Windows: 145. 윈도우 10 빌드 17046부터 WSL에서 백그라운드 작업 지원 [5]
11392정성태12/7/201717989개발 환경 구성: 341. openSUSE에 닷넷 코어 설치
11391정성태12/7/201720849개발 환경 구성: 340. WSL을 이용해 윈도우 PC 1대에서 openSUSE 응용 프로그램을 Visual Studio로 개발하는 방법 [1]
11390정성태12/7/201729487개발 환경 구성: 339. WSL을 이용해 윈도우 PC 1대에서 Linux 응용 프로그램을 Visual Studio로 개발하는 방법 [6]
11389정성태12/7/201718149오류 유형: 440. .NET Core 오류 - 0x80131620 Unable to load DLL 'libuv'
11388정성태12/6/201721795개발 환경 구성: 338. WSL 또는 Ubuntu에 닷넷 코어 설치 [3]
11387정성태12/6/201722109오류 유형: 439. 이벤트 로그 - Data Sharing Service 서비스의 %%3239247874 오류 메시지
11386정성태12/5/201717651오류 유형: 438. Hyper-V - '...' failed to add device 'Virtual CD/DVD Disk'
11385정성태12/5/201730767VC++: 121. DXGI를 이용한 윈도우 화면 캡처 소스 코드(Visual C++) [16]파일 다운로드1
11384정성태12/5/201720069오류 유형: 437. Visual C++ - Cannot open include file: 'SDKDDKVer.h'
11383정성태12/4/201723208디버깅 기술: 110. 비동기 코드 실행 중 예외로 인한 ASP.NET 프로세스 비정상 종료 현상 [1]
11382정성태12/4/201721798오류 유형: 436. System.Data.SqlClient.SqlException (0x80131904): Connection Timeout Expired 예외 발생 시 "[Pre-Login] initialization=48; handshake=1944;" 값의 의미
11381정성태11/30/201718192.NET Framework: 702. 한글이 포함된 바이트 배열을 나눈 경우 한글이 깨지지 않도록 다시 조합하는 방법(두 번째 이야기)파일 다운로드1
11380정성태11/30/201718235디버깅 기술: 109. windbg - (x64에서의 인자 값 추적을 이용한) Thread.Abort 시 대상이 되는 스레드를 식별하는 방법
11379정성태11/30/201718966오류 유형: 435. System.Web.HttpException - Session state has created a session id, but cannot save it because the response was already flushed by the application.
... 91  92  93  94  95  96  97  98  99  100  [101]  102  103  104  105  ...