Microsoft MVP성태의 닷넷 이야기
VC++: 132. enum 정의를 C++11의 enum class로 바꿀 때 유의할 사항 [링크 복사], [링크+제목 복사],
조회: 19499
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

enum 정의를 C++11의 enum class로 바꿀 때 유의할 사항

기존 enum의 명확한 단점은 다음의 소스 코드로 알 수 있습니다.

enum TestMode
{
    OFF,
};

enum MyMode
{
    OFF, // Error C2365 'OFF': redefinition; previous definition was 'enumerator'
};

이 때문에, 어쩔 수 없이 Prefix를 정의해 함께 사용해야만 했습니다.

enum TestMode
{
    TM_OFF,
};

enum MyMode
{
    MM_OFF,
};

윈도우 헤더 파일에 정의된 수많은 enum 타입들의 상수가 왜 타입명을 함께 붙이고 정의하는지,

// windef.h

typedef enum DPI_AWARENESS {
    DPI_AWARENESS_INVALID           = -1,
    DPI_AWARENESS_UNAWARE           = 0,
    DPI_AWARENESS_SYSTEM_AWARE      = 1,
    DPI_AWARENESS_PER_MONITOR_AWARE = 2
} DPI_AWARENESS;

바로 이런 이유 때문이었던 것입니다.




그러다가, C++11 표준에서 "enum class"가 나와 더 이상 접미사가 필요 없게 되었습니다.

enum class TestMode
{
    ON,
    OFF,
};

enum class MyMode
{
    ON,
    OFF
};

int main()
{
    MyMode mode1 = MyMode::ON;
    MyMode mode2 = ON; // 컴파일 오류
}

하지만, 그래도 내부적으로는 "#define"과의 충돌을 해결하지 못해 다음과 같은 상황에서는 컴파일 오류가 발생합니다.

#define ON 100

enum class MyMode
{
    ON, // Error C2059 syntax error: 'constant'
    OFF
};

물론 이해는 됩니다. "ON"이라는 리터럴 자체를 #define 정의에 의해 모조리 치환하기 때문에 결국 다음과 같이 번역되므로 오류가 발생하는 것이 당연합니다.

#define ON 100

enum class MyMode
{
    100, // Error C2059 syntax error: 'constant'
    OFF
};

따라서 기존 헤더 파일에 같은 상수명이 정의되어 있다면 접미사를 붙이거나 명시적으로 #undef으로 해결할 수밖에 없습니다.

enum class TestMode
{
    TM_ON,
    OFF,
};

enum class MyMode
{
 #undef ON
    ON,
    OFF
};




enum class의 또 하나 문제점(?)이 있다면, 형식 안정성으로 인해 기존 enum이 int 타입 연산을 자연스럽게 할 수 있었던 것을 못하게 막는다는 점입니다. 즉, 다음과 같은 식의 비트 플래그 연산들이 enum class 사용 후부터 모두 오류가 발생하게 됩니다.

int main()
{
    MyMode mode1 = MyMode::ON;
    MyMode mode2 = MyMode::OFF;

    // Error C2676 binary '&': 'MyMode' does not define this operator or a conversion to a type acceptable to the predefined operator
    if ((mode1 & mode2) == mode2)
    {
    }
}

이 문제를 해결하려면 명시적인 int 형변환을 하거나,

if (((int)mode1 & (int)mode2) == (int)mode2)

"(int)" 형변환 따위의 코드 수정 없이 하고 싶다면, 좀 더 우아하게는 다음과 같이 연산자 재정의를 추가하면 됩니다.

inline MyMode operator&(MyMode& l, MyMode &r)
{
    return (MyMode)((int)l & (int)r);
}

inline MyMode operator|(MyMode& l, MyMode& r)
{
    return (MyMode)((int)l | (int)r);
}

멋스럽게 & 연산자를 추가해봤지만 저런 경우 다음과 같은 식의 상황에서는 컴파일 오류가 발생하므로,

auto GetFlags() -> MyMode
{
    return mode;
}

int main()
{
    MyMode mode = MyMode::ON;
    MyMode mode2 = MyMode::OFF;

    if ((mode | mode2) == mode2)
    {
        mode = MyMode::ON;
    }

    char* ptr = nullptr;

    // Error C2678 binary '&': no operator found which takes a left-hand operand of type 'MyMode' (or there is no acceptable conversion)
    if ((GetFlags() & mode2) == mode2)
    {
        mode = MyMode::ON;
    }
}

이 상황을 해결하기 위해 const 등의 좀 더 복잡한 코드를 만들 수도 있지만,

auto GetFlags() -> const MyMode&
{
    return mode;
}

inline MyMode operator&(const MyMode& l, MyMode &r)
{
    return (MyMode)((int)l & (int)r);
}

어차피 enum이 int 범위 내의 값이라는 특성을 감안하면 굳이 const로 만들지 않아도 된다는 점과, 오히려 ref를 받아 변경할 수 있도록 하는 것이 아니라면 애당초 그냥 아무 처리 없이 사용하는 것이 더 나은 선택일 것입니다.

inline MyMode operator&(MyMode l, MyMode r)
{
    return (MyMode)((int)l & (int)r);
}

inline MyMode operator|(MyMode l, MyMode r)
{
    return (MyMode)((int)l | (int)r);
}

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




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







[최초 등록일: ]
[최종 수정일: 6/7/2019]

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)
13768정성태10/15/20245396C/C++: 179. C++ - _O_WTEXT, _O_U16TEXT, _O_U8TEXT의 Unicode stream 모드파일 다운로드2
13767정성태10/14/20244771오류 유형: 929. bpftrace 수행 시 "ERROR: Could not resolve symbol: /proc/self/exe:BEGIN_trigger"
13766정성태10/14/20244553C/C++: 178. C++ - 파일에 대한 Text 모드의 "translated" 동작파일 다운로드1
13765정성태10/12/20245266오류 유형: 928. go build 시 "package maps is not in GOROOT" 오류
13764정성태10/11/20245631Linux: 85. Ubuntu - 원하는 golang 버전 설치
13763정성태10/11/20244985Linux: 84. WSL / Ubuntu 20.04 - bpftool 설치
13762정성태10/11/20245010Linux: 83. WSL / Ubuntu 22.04 - bpftool 설치
13761정성태10/11/20244915오류 유형: 927. WSL / Ubuntu - /usr/include/linux/types.h:5:10: fatal error: 'asm/types.h' file not found
13760정성태10/11/20245450Linux: 82. Ubuntu - clang 최신(stable) 버전 설치
13759정성태10/10/20246366C/C++: 177. C++ - 자유 함수(free function) 및 주소 지정 가능한 함수(addressable function) [6]
13758정성태10/8/20245579오류 유형: 926. dotnet tools를 sudo로 실행하는 경우 command not found
13757정성태10/8/20245521닷넷: 2306. Linux - dotnet tool의 설치 디렉터리가 PATH 환경변수에 자동 등록이 되는 이유
13756정성태10/8/20245627오류 유형: 925. ssh로 docker 접근을 할 때 "... malformed HTTP status code ..." 오류 발생
13755정성태10/7/20246026닷넷: 2305. C# 13 - (9) 메서드 바인딩의 우선순위를 지정하는 OverloadResolutionPriority 특성 도입 (Overload resolution priority)파일 다운로드1
13754정성태10/4/20245579닷넷: 2304. C# 13 - (8) 부분 메서드 정의를 속성 및 인덱서에도 확대파일 다운로드1
13753정성태10/4/20245594Linux: 81. Linux - PATH 환경변수의 적용 규칙
13752정성태10/2/20246281닷넷: 2303. C# 13 - (7) ref struct의 interface 상속 및 제네릭 제약으로 사용 가능 [6]파일 다운로드1
13751정성태10/2/20245407C/C++: 176. C/C++ - ARM64로 포팅할 때 유의할 점
13750정성태10/1/20245293C/C++: 175. C++ - WinMain/wWinMain 호출 전의 CRT 초기화 단계
13749정성태9/30/20245536닷넷: 2302. C# - ssh-keygen으로 생성한 Private Key와 Public Key 연동파일 다운로드1
13748정성태9/29/20245743닷넷: 2301. C# - BigInteger 타입이 byte 배열로 직렬화하는 방식
13747정성태9/28/20245592닷넷: 2300. C# - OpenSSH의 공개키 파일에 대한 "BEGIN OPENSSH PUBLIC KEY" / "END OPENSSH PUBLIC KEY" PEM 포맷파일 다운로드1
13746정성태9/28/20245688오류 유형: 924. Python - LocalProtocolError("Illegal header value ...")
13745정성태9/28/20245550Linux: 80. 리눅스 - 실행 중인 프로세스 내부의 환경변수 설정을 구하는 방법 (lldb)
13744정성태9/27/20245981닷넷: 2299. C# - Windows Hello 사용자 인증 다이얼로그 표시하기파일 다운로드1
13743정성태9/26/20246427닷넷: 2298. C# - Console 프로젝트에서의 await 대상으로 Main 스레드 활용하는 방법 [1]
1  2  3  4  5  6  [7]  8  9  10  11  12  13  14  15  ...