Microsoft MVP성태의 닷넷 이야기
VC++: 97. C++ 템플릿 remove_pointer, enable_if, is_pointer 사용 예제 [링크 복사], [링크+제목 복사],
조회: 14365
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

C++ 템플릿 remove_pointer, enable_if, is_pointer 사용 예제

처음 해결하고 싶었던 문제는 다음과 같은 식의 코드였습니다.

#include "stdafx.h"

using namespace std;
#include <map>

struct MyType { MyType() { } };

template <typename _Key, typename _Value>
class MapTS
{
public:
    MapTS()
    {
    }

    void SetNew(_Key key)
    {
        _Value v = new _Value();
        m_map.insert(pair<_Key, _Value>(key, v));
    }

private:
    map<_Key, _Value> m_map;
};

int main()
{
    MapTS<int, MyType *> m2;
    m2.SetNew(6);

    return 0;
}

위의 코드를 컴파일하면 다음과 같은 오류가 발생합니다.

Error C2440 'initializing': cannot convert from 'MyType **' to 'MyType *'

당연합니다. "_Value v = new _Value();" 코드가 다음과 같이 바뀌기 때문입니다.

MyType *v = new MyType *();

이를 해결하기 위해서는 포인터를 하나 제거한 타입을 구해야 하는데요. 이때 사용할 수 있는 것이 바로 remove_pointer 구문입니다. 따라서, 다음과 같이 해주면 해결할 수 있습니다.

template <typename _Key, typename _Value>
class MapTS
{
    typedef typename remove_pointer<_Value>::type _StructValue;

public:
    MapTS()
    {
    }

    void SetNew(_Key key)
    {
        _StructValue *v = new _StructValue();
        m_map.insert(pair<_Key, _Value>(key, v));
    }

private:
    map<_Key, _Value> m_map;
};

그런데, 동일한 코드로 포인터 없는 형태로도 사용하고 싶어졌습니다.

MapTS<int, MyType> m1;
m1.SetNew(5);

이렇게 사용하면 다음과 같은 오류가 발생하게 됩니다.

Error C2440 '<function-style-cast>': cannot convert from 'initializer list' to 'std::pair<_Key,_Value>'

역시 당연합니다. "MyType"을 주었기 때문에 m_map 변수의 타입이 "map<int, MyType>"으로 바뀌었고 SetNew 함수에서 포인터를 밀어넣으려니 오류가 발생하는 것입니다. 따라서, 포인터 없는 구문이 동작하려면 SetNew 함수가 다음과 같이 바뀌어야 합니다.

void SetNew(_Key key)
{
    _Value v = _Value();
    m_map.insert(pair<_Key, _Value>(key, v));
}

이를 해결하고 싶다면, 포인터 있는 경우와 없는 경우를 구분하고 그에 따른 템플릿 특수화를 해야 합니다. 이를 위해 is_pointer와 enable_if를 다음과 같이 적용해 줄 수 있습니다. (enable_if의 존재를 회사 동료가 알려줬는데... 전혀 예상치 못한 확장이었습니다. ^^;)

template <typename _Key, typename _Value>
class MapTS
{
    typedef typename remove_pointer<_Value>::type _StructValue;

public:
    MapTS() { }

    template<typename T = _Value>
    void SetNew(_Key key, typename enable_if< is_pointer<T>::value, T >::type value = nullptr)
    {
        _StructValue *v = new _StructValue();
        m_map.insert(pair<_Key, _Value>(key, v));
    }

    template<typename T = _Value>
    void SetNew(_Key key, typename enable_if< !is_pointer<T>::value, T >::type value = _Value())
    {
        m_map.insert(pair<_Key, _Value>(key, _Value()));
    }

private:
    map<_Key, _Value> m_map;
};

int main()
{
    MapTS<int, MyType> m1;
    m1.SetNew(5);

    MapTS<int, MyType *> m2;
    m2.SetNew(6);

    return 0;
}

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




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







[최초 등록일: ]
[최종 수정일: 5/31/2016]

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

비밀번호

댓글 작성자
 




... 46  47  48  49  50  51  52  53  54  55  [56]  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
12227정성태6/12/202016047오류 유형: 616. Visual Studio의 느린 업데이트 속도에 대한 원인 분석 [5]
12226정성태6/11/202013344개발 환경 구성: 493. OpenVPN의 네트워크 구성 [4]파일 다운로드1
12225정성태6/11/202012332개발 환경 구성: 492. 윈도우에 OpenVPN 설치 - 클라이언트 측 구성
12224정성태6/11/202020194개발 환경 구성: 491. 윈도우에 OpenVPN 설치 - 서버 측 구성 [1]
12223정성태6/9/202014196.NET Framework: 908. C# - Source Generator 소개 [10]파일 다운로드2
12222정성태6/3/202010100VS.NET IDE: 146. error information: "CryptQueryObject" (-2147024893/0x80070003)
12221정성태6/3/20209867Windows: 170. 비어 있지 않은 디렉터리로 symbolic link(junction) 연결하는 방법
12220정성태6/3/202012261.NET Framework: 907. C# DLL로부터 TLB 및 C/C++ 헤더 파일(TLH)을 생성하는 방법
12219정성태6/1/202011407.NET Framework: 906. C# - lock (this), lock (typeof(...))를 사용하면 안 되는 이유파일 다운로드1
12218정성태5/27/202011332.NET Framework: 905. C# - DirectX 게임 클라이언트 실행 중 키보드 입력을 감지하는 방법 [3]
12217정성태5/24/20209804오류 유형: 615. Transaction count after EXECUTE indicates a mismatching number of BEGIN and COMMIT statements. Previous count = 0, current count = 1.
12216정성태5/15/202012942.NET Framework: 904. USB/IP PROJECT를 이용해 C#으로 USB Keyboard 가상 장치 만들기 [14]파일 다운로드1
12215정성태5/12/202017953개발 환경 구성: 490. C# - (Wireshark의) USBPcap을 이용한 USB 패킷 모니터링 [10]파일 다운로드1
12214정성태5/5/202010283개발 환경 구성: 489. 정식 인증서가 있는 경우 Device Driver 서명하는 방법 (2) - UEFI/SecureBoot [1]
12213정성태5/3/202011929개발 환경 구성: 488. (User-mode 코드로 가상 USB 장치를 만들 수 있는) USB/IP PROJECT 소개
12212정성태5/1/20209574개발 환경 구성: 487. UEFI / Secure Boot 상태인지 확인하는 방법
12211정성태4/27/202011918개발 환경 구성: 486. WSL에서 Makefile로 공개된 리눅스 환경의 C/C++ 소스 코드 빌드
12210정성태4/20/202012357.NET Framework: 903. .NET Framework의 Strong-named 어셈블리 바인딩 (1) - app.config을 이용한 바인딩 리디렉션 [1]파일 다운로드1
12209정성태4/13/202010444오류 유형: 614. 리눅스 환경에서 C/C++ 프로그램이 Segmentation fault 에러가 발생한 경우 (2)
12208정성태4/12/20209908Linux: 29. 리눅스 환경에서 C/C++ 프로그램이 Segmentation fault 에러가 발생한 경우
12207정성태4/2/20208879스크립트: 19. Windows PowerShell의 NonInteractive 모드
12206정성태4/2/202011147오류 유형: 613. 파일 잠금이 바로 안 풀린다면? - The process cannot access the file '...' because it is being used by another process.
12205정성태4/2/20208597스크립트: 18. Powershell에서는 cmd.exe의 명령어를 지원하진 않습니다.
12204정성태4/1/20208376스크립트: 17. Powershell 명령어에 ';' (semi-colon) 문자가 포함된 경우
12203정성태3/18/202010407오류 유형: 612. warning: 'C:\ProgramData/Git/config' has a dubious owner: '...'.
12202정성태3/18/202013032개발 환경 구성: 486. .NET Framework 프로젝트를 위한 GitLab CI/CD Runner 구성
... 46  47  48  49  50  51  52  53  54  55  [56]  57  58  59  60  ...