Microsoft MVP성태의 닷넷 이야기
닷넷: 2359. C# 14 - (10) 복합 대입 연산자의 오버로드 지원 [링크 복사], [링크+제목 복사],
조회: 3899
글쓴 사람
정성태 (seongtaejeong at gmail.com)
홈페이지
첨부 파일

(시리즈 글이 12개 있습니다.)
닷넷: 2342. C# 14 - (취소된 글)
; https://www.sysnet.pe.kr/2/0/13970

닷넷: 2343. C# 14 - (1) 속성 구문에서 문맥 키워드로 추가되는 field 예약어
; https://www.sysnet.pe.kr/2/0/13971

닷넷: 2346. C# 14 - (2) Span 타입과 배열 간의 암시적 형변환
; https://www.sysnet.pe.kr/2/0/13974

닷넷: 2347. C# 14 - (3) 형식 인자가 없는 제네릭 타입의 nameof 지원
; https://www.sysnet.pe.kr/2/0/13975

닷넷: 2349. C# 14 - (4) 문자열 리터럴을 utf-8 인코딩으로 저장
; https://www.sysnet.pe.kr/2/0/13977

닷넷: 2350. C# 14 - (5) 람다 매개 변수에 접근자가 있는 경우에도 타입 생략 가능
; https://www.sysnet.pe.kr/2/0/13986

닷넷: 2351. C# 14 - (6) event와 생성자에도 partial 메서드 적용
; https://www.sysnet.pe.kr/2/0/13987

닷넷: 2354. C# 14 - (7) 확장 메서드에 정적 메서드와 속성 지원을 위한 전용 구문 추가
; https://www.sysnet.pe.kr/2/0/13998

닷넷: 2355. C# 14 - (8) null 조건부 연산자 개선 - 대입문에도 사용 가능
; https://www.sysnet.pe.kr/2/0/13999

닷넷: 2357. C# 14 - (9) 새로운 지시자 추가 (Ignored directives)
; https://www.sysnet.pe.kr/2/0/14003

닷넷: 2359. C# 14 - (10) 복합 대입 연산자의 오버로드 지원
; https://www.sysnet.pe.kr/2/0/14008

닷넷: 2360. C# 14 - (11) Expression Tree에 선택적 인수와 명명된 인수 허용
; https://www.sysnet.pe.kr/2/0/14009




C# 14 - (10) 복합 대입 연산자의 오버로드 지원

(2025-08-26 기준) 이번 글의 실습을 하려면 Visual Studio 2022 Preview 버전과 .NET 10 SDK Preview 버전을 함께 설치하고, csproj의 TargetFramework를 "net10.0"으로 설정해야 합니다.

<Project Sdk="Microsoft.NET.Sdk">

    <PropertyGroup>
        <OutputType>Exe</OutputType>
        <TargetFramework>net10.0</TargetFramework>
        <ImplicitUsings>enable</ImplicitUsings>
        <Nullable>enable</Nullable>
        <LangVersion>preview</LangVersion>
    </PropertyGroup>

</Project>




C# 13 이전에는 복합 대입 연산자에 대한 "연산자 오버로드(operator overload)"를 지원하지 않았습니다. 예를 들어 "a += b"와 같은 식에서 "+=" 연산자 자체를 오버로드할 수 없었는데요, C#은 대신 이 문제를 "a = a + b"로 변환해 결국 "+" 연산자의 오버로드 버전을 사용한 대입으로 풀어 처리했습니다.

자, 그럼 예제 코드와 함께 이야기를 이어가 볼까요? ^^ 아래는 + 연산자를 재정의한 예제입니다.

Kilogram a = new Kilogram(5);
Kilogram b = new Kilogram(10);

Kilogram c = a + b;
Console.WriteLine(c); // 출력 결과: 15 kg

public class Kilogram
{
    decimal mass;

    public Kilogram(decimal mass)
    {
        this.mass = mass;
    }

    // '+' 연산자 오버로드
    public static Kilogram operator +(Kilogram a, Kilogram b)
    {
        return new Kilogram(a.mass + b.mass);
    }

    public override string ToString() => $"{mass} kg";
}

전형적인 연산자 오버로드 예제인데요, 그런데 여기에 살짝 아쉬운 점이 하나 있습니다. 위의 경우처럼 "c = a + b" 형태로 사용하는 식에서는 문제가 없지만, "a += b"처럼 복합 대입문(compound assignment statement)으로 사용하는 경우에는 비효율적인 코드가 생성된다는 점입니다.

왜냐하면 "a += b"는 결국 a 인스턴스의 값을 변경하는 것이므로 굳이 "new Kilogram(...)"의 새로운 객체를 반환할 필요가 없기 때문입니다. C# 14는 바로 이런 문제를 해결하기 위해 복합 대입 연산자에 대한 오버로드 지원을 추가합니다.

이렇게 오버로드 지원을 추가하기로 결정했다면, 이제 어떤 식으로 구현해야 하느냐에 대한 문제가 남습니다. 위의 코드에서 해결하고자 하는 것은 "new Kilogram(...)" 코드로 할당하는 것을 없애고 싶다는 건데요, 마이크로소프트는 이 문제를 해결하기 위해 "정적 메서드"가 아닌 "인스턴스 메서드"로도 연산자 오버로드를 정의하는 방식을 택합니다.

따라서 C# 14의 복합 대입 연산자는 오버로드 시 아래와 같은 새로운 형태의 문법으로 정의하게 됩니다.

public void operator 복합_대입_연산자(타입1 변수명1) { .... }

// 1. 인스턴스 메서드 형태
// 2. 반환값은 없으므로 void
// 3. 매개변수는 하나
// 4. 연산자 오버로드 메서드 이름은 'operator 연산자' 형태

위의 조건을 만족하는 C# 14의 문법으로 클래스를 정의하면 다음과 같습니다.

Kilogram a = new Kilogram(5);
Kilogram b = new Kilogram(10);

a += b;
Console.WriteLine(a); // 출력 결과: 15 kg

public class Kilogram
{
    decimal mass;

    public Kilogram(decimal mass) { this.mass = mass; }

    // 인스턴스 메서드 유형의 '+=' 복합 대입 연산자 오버로드 (new가 없으므로 새로운 개체 생성이 없음!)
    public void operator +=(Kilogram b)
    {
        this.mass += b.mass;
    }

    public override string ToString() => $"{mass} kg";
}

보는 바와 같이 새로운 개체를 생성하지 않고 this.mass 값을 직접 변경하는 형태로 구현하기 때문에 불필요한 GC Heap 할당이 발생하지 않습니다.




인스턴스 메서드 형태의 연산자 오버로드 버전이 기왕에 도입되었으니, 이것을 통해 다른 연산자의 오버로드도 함께 개선할 수 있다면 더 좋겠지요. ^^ 바로 이에 부합하는 것이 증감 연산자입니다.

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

증감 연산자가 식 내에서 사용한다면 문제가 좀 복잡해지는데요, 간단하게 단문으로만 존재한다면,

Kilogram a = new Kilogram(1);
a ++; // 어차피 a의 값이 변경
++ a; // 어차피 a의 값이 변경

/*
// 기존의 정적 오버로드 메서드
public static Kilogram operator ++(Kilogram a)
{
    return new Kilogram(a.mass + 1);
}
*/

굳이 static 메서드로 정의한 오버로드 버전을 사용할 필요가 없습니다. 즉, 위와 같은 경우에도 인스턴스 버전의 오버로드를 정의하는 것이 성능상 장점이 됩니다.

// C# 14부터 지원

public void operator ++()
{
    this.mass++;
}

public static Kilogram operator ++(Kilogram a)
{
    return new Kilogram(a.mass + 1);
}

위와 같이 증감 연산자가 단문으로 사용한 경우 이외에도, 수식 내에서 사용했다면 전위/후위냐에 따라 바인딩이 달라지는 차이점도 있습니다. 예를 들어 아래와 같은 경우,

Kilogram a = new Kilogram(1);

// 후위 증감 연산자 사용 시, 정적 메서드 유형의 오버로드가 호출됨
Kilogram b = a++;

// 전위 증감 연산자 사용 시, 인스턴스 메서드 유형의 오버로드가 호출됨
Kilogram c = ++a;

전위 증감 연산자의 호출 결과가 곧 a 자신이 바뀐 이후이므로 인스턴스 메서드 유형의 오버로드를 호출하도록 C# 컴파일러가 알아서 바인딩을 바꿔줍니다.

이외에도 증감 연산자는 class와 struct에 대한 오버로드 선택 시 차이점을 가집니다. 이에 대해서는 다음의 글에서 자세하게 예제 코드와 함께 다루고 있으니 참고하시면 됩니다.

Prefix increment and decrement operators
; https://learn.microsoft.com/en-us/dotnet/csharp/language-reference/proposals/csharp-14.0/user-defined-compound-assignment#prefix-increment-and-decrement-operators

Postfix increment and decrement operators
; https://learn.microsoft.com/en-us/dotnet/csharp/language-reference/proposals/csharp-14.0/user-defined-compound-assignment#prefix-increment-and-decrement-operators>




정리해 보면, 연산자 오버로드를 기존 C# 13처럼 정적 메서드 유형으로만 정의해도 좋습니다. 단지 여러분의 타입을 대단위 수학 연산에서 사용한다면 성능을 고려해 인스턴스 메서드 유형의 오버로드를 추가하는 것이 권장된다는 정도로 가볍게 받아들이면 됩니다.

어차피 우리가 정의만 해주면 C# 컴파일러가 알아서 최적의 오버로드 메서드를 바인딩해 줄 것이므로!




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







[최초 등록일: ]
[최종 수정일: 8/26/2025]

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)
12136정성태2/6/202022777Windows: 168. Windows + S(또는 Q)로 뜨는 작업 표시줄의 검색 바가 동작하지 않는 경우
12135정성태2/6/202028416개발 환경 구성: 468. Nuget 패키지의 로컬 보관 폴더를 옮기는 방법 [2]
12134정성태2/5/202025776.NET Framework: 884. eBEST XingAPI의 C# 래퍼 버전 - XingAPINet Nuget 패키지 [5]파일 다운로드1
12133정성태2/5/202023560디버깅 기술: 161. Windbg 환경에서 확인해 본 .NET 메서드 JIT 컴파일 전과 후 - 두 번째 이야기
12132정성태1/28/202026769.NET Framework: 883. C#으로 구현하는 Win32 API 후킹(예: Sleep 호출 가로채기) [1]파일 다운로드1
12131정성태1/27/202024983개발 환경 구성: 467. LocaleEmulator를 이용해 유니코드를 지원하지 않는(한글이 깨지는) 프로그램을 실행하는 방법 [1]
12130정성태1/26/202022635VS.NET IDE: 142. Visual Studio에서 windbg의 "Open Executable..."처럼 EXE를 직접 열어 디버깅을 시작하는 방법
12129정성태1/26/202029721.NET Framework: 882. C# - 키움 Open API+ 사용 시 Registry 등록 없이 KHOpenAPI.ocx 사용하는 방법 [3]
12128정성태1/26/202024125오류 유형: 591. The code execution cannot proceed because mfc100.dll was not found. Reinstalling the program may fix this problem.
12127정성태1/25/202020806.NET Framework: 881. C# DLL에서 제공하는 Win32 export 함수의 내부 동작 방식(VT Fix up Table)파일 다운로드1
12126정성태1/25/202023217.NET Framework: 880. C# - PE 파일로부터 IMAGE_COR20_HEADER 및 VTableFixups 테이블 분석파일 다운로드1
12125정성태1/24/202021835VS.NET IDE: 141. IDE0019 - Use pattern matching
12124정성태1/23/202023594VS.NET IDE: 140. IDE1006 - Naming rule violation: These words must begin with upper case characters: ...
12123정성태1/23/202024143웹: 39. Google Analytics - gtag 함수를 이용해 페이지 URL 수정 및 별도의 이벤트 생성 방법 [2]
12122정성태1/20/202021055.NET Framework: 879. C/C++의 UNREFERENCED_PARAMETER 매크로를 C#에서 우회하는 방법(IDE0060 - Remove unused parameter '...')파일 다운로드1
12121정성태1/20/202019762VS.NET IDE: 139. Visual Studio - Error List: "Could not find schema information for the ..."파일 다운로드1
12120정성태1/19/202024084.NET Framework: 878. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 네 번째 이야기(IL 코드로 직접 구현)파일 다운로드1
12119정성태1/17/202024661디버깅 기술: 160. Windbg 확장 DLL 만들기 (3) - C#으로 만드는 방법
12118정성태1/17/202025338개발 환경 구성: 466. C# DLL에서 Win32 C/C++처럼 dllexport 함수를 제공하는 방법 - 세 번째 이야기 [1]
12117정성태1/15/202023631디버깅 기술: 159. C# - 디버깅 중인 프로세스를 강제로 다른 디버거에서 연결하는 방법파일 다운로드1
12116정성태1/15/202025400디버깅 기술: 158. Visual Studio로 디버깅 시 sos.dll 확장 명령어를 (비롯한 windbg의 다양한 기능을) 수행하는 방법
12115정성태1/14/202025203디버깅 기술: 157. C# - PEB.ProcessHeap을 이용해 디버깅 중인지 확인하는 방법파일 다운로드1
12114정성태1/13/202025996디버깅 기술: 156. C# - PDB 파일로부터 심벌(Symbol) 및 타입(Type) 정보 열거 [1]파일 다운로드3
12113정성태1/12/202026657오류 유형: 590. Visual C++ 빌드 오류 - fatal error LNK1104: cannot open file 'atls.lib' [1]
12112정성태1/12/202020709오류 유형: 589. PowerShell - 원격 Invoke-Command 실행 시 "WinRM cannot complete the operation" 오류 발생
12111정성태1/12/202025041디버깅 기술: 155. C# - KernelMemoryIO 드라이버를 이용해 실행 프로그램을 숨기는 방법(DKOM: Direct Kernel Object Modification) [16]파일 다운로드1
... [76]  77  78  79  80  81  82  83  84  85  86  87  88  89  90  ...