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

(시리즈 글이 11개 있습니다.)
닷넷: 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




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

비밀번호

댓글 작성자
 




1  2  3  4  5  6  [7]  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13859정성태1/9/20257432디버깅 기술: 215. Windbg - syscall 이후 실행되는 KiSystemCall64 함수 및 SSDT 디버깅
13858정성태1/8/20257414개발 환경 구성: 738. PowerShell - 원격 호출 시 "powershell.exe"가 아닌 "pwsh.exe" 환경으로 명령어를 실행하는 방법
13857정성태1/7/20257872C/C++: 187. Golang - 콘솔 응용 프로그램을 Linux 데몬 서비스를 지원하도록 변경파일 다운로드1
13856정성태1/6/20255795디버깅 기술: 214. Windbg - syscall 단계까지의 Win32 API 호출 (예: Sleep)
13855정성태12/28/20248364오류 유형: 941. Golang - os.StartProcess() 사용 시 오류 정리
13854정성태12/27/20248186C/C++: 186. Golang - 콘솔 응용 프로그램을 NT 서비스를 지원하도록 변경파일 다운로드1
13853정성태12/26/20246328디버깅 기술: 213. Windbg - swapgs 명령어와 (Ring 0 커널 모드의) FS, GS Segment 레지스터
13852정성태12/25/20248395디버깅 기술: 212. Windbg - (Ring 3 사용자 모드의) FS, GS Segment 레지스터파일 다운로드1
13851정성태12/23/20246682디버깅 기술: 211. Windbg - 커널 모드 디버깅 상태에서 사용자 프로그램을 디버깅하는 방법
13850정성태12/23/20248586오류 유형: 940. "Application Information" 서비스를 중지한 경우, "This file does not have an app associated with it for performing this action."
13849정성태12/20/20248705디버깅 기술: 210. Windbg - 논리(가상) 주소를 Segmentation을 거쳐 선형 주소로 변경
13848정성태12/18/20247668디버깅 기술: 209. Windbg로 알아보는 Prototype PTE파일 다운로드2
13847정성태12/18/20247480오류 유형: 939. golang - 빌드 시 "unknown directive: toolchain" 오류 빌드 시 이런 오류가 발생한다면?
13846정성태12/17/20248386디버깅 기술: 208. Windbg로 알아보는 Trans/Soft PTE와 2가지 Page Fault 유형파일 다운로드1
13845정성태12/16/20246332디버깅 기술: 207. Windbg로 알아보는 PTE (_MMPTE)
13844정성태12/14/20249433디버깅 기술: 206. Windbg로 알아보는 PFN (_MMPFN)파일 다운로드1
13843정성태12/13/20246756오류 유형: 938. Docker container 내에서 빌드 시 error MSB3021: Unable to copy file "..." to "...". Access to the path '...' is denied.
13842정성태12/12/20246981디버깅 기술: 205. Windbg - KPCR, KPRCB
13841정성태12/11/20247901오류 유형: 937. error MSB4044: The "ValidateValidArchitecture" task was not given a value for the required parameter "RemoteTarget"
13840정성태12/11/20246775오류 유형: 936. msbuild - Your project file doesn't list 'win' as a "RuntimeIdentifier"
13839정성태12/11/20248598오류 유형: 936. msbuild - error CS1617: Invalid option '12.0' for /langversion. Use '/langversion:?' to list supported values.
13838정성태12/4/20247974오류 유형: 935. Windbg - Breakpoint 0's offset expression evaluation failed.
13837정성태12/3/20249123디버깅 기술: 204. Windbg - 윈도우 핸들 테이블 (3) - Windows 10 이상인 경우
13836정성태12/3/20246631디버깅 기술: 203. Windbg - x64 가상 주소를 물리 주소로 변환 (페이지 크기가 2MB인 경우)
13835정성태12/2/20249088오류 유형: 934. Azure - rm: cannot remove '...': Directory not empty
13834정성태11/29/20249002Windows: 275. C# - CUI 애플리케이션과 Console 윈도우 (Windows 10 미만의 Classic Console 모드인 경우) [1]파일 다운로드1
1  2  3  4  5  6  [7]  8  9  10  11  12  13  14  15  ...