Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

G++ - 템플릿 클래스의 iterator 코드 사용에서 발생하는 컴파일 오류

G++ 컴파일러로 다음의 코드를 컴파일 하면,

#include <iostream>
#include <vector>
#include <map>

typedef char BYTE;
typedef int _int;
typedef std::vector<BYTE> _byteArrayHolder;
typedef long _long;

template <class T, class V>
class KeepAliveReceiver
{
};

template<class Receiver, class ErrorAction, _int MAX_CONTENT_SIZE = 8192>
class MultiPartByteArrayReceiver : public KeepAliveReceiver< MultiPartByteArrayReceiver<Receiver, ErrorAction>, ErrorAction>
{
    class MuliPartByteArray
    {
        _int key;
        std::vector<BYTE> byteHolders;
        _long totalSize;

    public:
        MuliPartByteArray(_int _key) : key(_key), byteHolders(), totalSize(0)
        {
        }

        void AddBytes(_byteArrayHolder bytes)
        {
            totalSize += bytes.size();
        }
    };

    typedef KeepAliveReceiver< MultiPartByteArrayReceiver<Receiver, ErrorAction, MAX_CONTENT_SIZE>, ErrorAction> Super;
    typedef MultiPartByteArrayReceiver<Receiver, ErrorAction, MAX_CONTENT_SIZE> This;

public:
    typedef std::map<_int, MuliPartByteArray>  MultiPartBytesMap;
    MultiPartBytesMap receivedMultiPartedBytes;

public:

    MultiPartByteArrayReceiver() : Super(), receivedMultiPartedBytes()
    {}

    void Test()
    {
        std::map<_int, MultiPartByteArray>::iterator iter = receivedMultiPartedBytes.find(1); // 컴파일 오류
    }
};

using namespace std;

int main()
{
    cout << "Hello World" << endl;

    MultiPartByteArrayReceiver<int, int> t;

    size_t size = t.receivedMultiPartedBytes.size();

   return 0;
}

이런 컴파일 오류가 발생합니다.

main.cpp: In member function void MultiPartByteArrayReceiver<Receiver, ErrorAction, MAX_CONTENT_SIZE>::Test():
main.cpp:60:28: error: MultiPartByteArray was not declared in this scope
             std::map<_int, MultiPartByteArray>::iterator iter = receivedMultiPartedBytes.find(1);
                            ^
main.cpp:60:46: error: template argument 2 is invalid
             std::map<_int, MultiPartByteArray>::iterator iter = receivedMultiPartedBytes.find(1);
                                              ^
main.cpp:60:46: error: template argument 4 is invalid
main.cpp:60:58: error: expected initializer before iter
             std::map<_int, MultiPartByteArray>::iterator iter = receivedMultiPartedBytes.find(1);
                                                          ^

아~~~ 이 현란한 C++ template 사용이여~~~~!

위의 컴파일 오류를 보여준 직장 동료의 우회 방법은 MultiPartByteArrayReceiver 클래스 내에 inner 클래스로 정의된 MuliPartByteArray를 바깥으로 빼내는 것이었습니다.

#include <iostream>
#include <vector>
#include <map>

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

class MuliPartByteArray
{
    // ... [생략]...
};


template<class Receiver, class ErrorAction, _int MAX_CONTENT_SIZE = 8192>
class MultiPartByteArrayReceiver : public KeepAliveReceiver< MultiPartByteArrayReceiver<Receiver, ErrorAction>, ErrorAction>
{
    // ... [생략]...
};

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

그런데... 사실 inner 클래스는 이유가 있기 때문에 inner 클래스가 된 것인데 그것을 바깥으로 빼내는 것은 그리 바람직해 보이지 않았습니다. 제가 제시한 해결책은 클래스의 전방 선언이었는데, 다음과 같이 처리하는 것이었습니다.

#include <iostream>
#include <vector>
#include <map>

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

class MuliPartByteArray;

template<class Receiver, class ErrorAction, _int MAX_CONTENT_SIZE = 8192>
class MultiPartByteArrayReceiver : public KeepAliveReceiver< MultiPartByteArrayReceiver<Receiver, ErrorAction>, ErrorAction>
{
    class MuliPartByteArray
    {
        // ... [생략]...
    };

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

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

이렇게 해주면 컴파일이 잘 됩니다. 직장 동료가 이유를 물었지만... 허허허~~~ 어찌 그것을 제가 알 수 있겠습니까? 단지 과거에 C++ template과 잠시 싸워 온 일종의 감각으로 전방 선언이 효력이 있을 거라는 생각을 했을 뿐입니다. (혹시, 명확한 이유를 알고 계신 분이 있다면 덧글 좀~~~~ ^^)

참고로, 이 예제의 코드는 Visual C++ 2013에서는 전방 선언 필요없이도 잘 컴파일됩니다.




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







[최초 등록일: ]
[최종 수정일: 10/23/2014]

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

비밀번호

댓글 작성자
 



2014-10-23 05시15분
[직장 동료 ] ㅋㅋㅋㅋ 알아내란 말이에욧! ㅋㅋ
[guest]
2014-10-28 01시34분
[Lyn] 주제넘게 글 하나 써봣습니다 ^^; http://lunapiece.net/Article/14007373
[guest]
2014-10-28 01시42분
@Lyn 오~~~ 멋져요. ^^ typename 예약어(및 class)가 그렇게도 들어간다는 것은 처음 알았군요. (제가 한창 C++ 사용하던 때의 문법으로는 도저히 설명이 안되는 구문이라는. ^^;)
정성태
2021-06-13 04시07분
[Lyn] 블로그를 옮겨서 혹시나 해서 글을 다시 달아둡니다

http://blog.lunapiece.net/posts/C++-Template-Inner-Class-Type/
[guest]
2023-05-31 11시17분
On creating (and using) a transforming iterator
; https://devblogs.microsoft.com/oldnewthing/20230523-00/?p=108233
정성태

... 16  17  18  [19]  20  21  22  23  24  25  26  27  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
13158정성태11/9/20225259오류 유형: 826. Workload definition 'wasm-tools' in manifest 'microsoft.net.workload.mono.toolchain' [...] conflicts with manifest 'microsoft.net.workload.mono.toolchain.net7'
13157정성태11/8/20225911.NET Framework: 2065. C# - Mutex의 비동기 버전파일 다운로드1
13156정성태11/7/20226831.NET Framework: 2064. C# - Mutex와 Semaphore/SemaphoreSlim 차이점파일 다운로드1
13155정성태11/4/20226341디버깅 기술: 183. TCP 동시 접속 (연결이 아닌) 시도를 1개로 제한한 서버
13154정성태11/3/20225825.NET Framework: 2063. .NET 5+부터 지원되는 GC.GetGCMemoryInfo파일 다운로드1
13153정성태11/2/20227108.NET Framework: 2062. C# - 코드로 재현하는 소켓 상태(SYN_SENT, SYN_RECV)
13152정성태11/1/20225738.NET Framework: 2061. ASP.NET Core - DI로 추가한 클래스의 초기화 방법 [1]
13151정성태10/31/20225868C/C++: 161. Windows 11 환경에서 raw socket 테스트하는 방법파일 다운로드1
13150정성태10/30/20225902C/C++: 160. Visual Studio 2022로 빌드한 C++ 프로그램을 위한 다른 PC에서 실행하는 방법
13149정성태10/27/20225846오류 유형: 825. C# - CLR ETW 이벤트 수신이 GCHeapStats_V1/V2에 대해 안 되는 문제파일 다운로드1
13148정성태10/26/20225811오류 유형: 824. msbuild 에러 - error NETSDK1005: Assets file '...\project.assets.json' doesn't have a target for 'net5.0'. Ensure that restore has run and that you have included 'net5.0' in the TargetFramew
13147정성태10/25/20224907오류 유형: 823. Visual Studio 2022 - Unable to attach to CoreCLR. The debugger's protocol is incompatible with the debuggee.
13146정성태10/24/20225755.NET Framework: 2060. C# - Java의 Xmx와 유사한 힙 메모리 최댓값 제어 옵션 HeapHardLimit
13145정성태10/21/20226033오류 유형: 822. db2 - Password validation for user db2inst1 failed with rc = -2146500508
13144정성태10/20/20225850.NET Framework: 2059. ClrMD를 이용해 윈도우 환경의 메모리 덤프로부터 닷넷 모듈을 추출하는 방법파일 다운로드1
13143정성태10/19/20226387오류 유형: 821. windbg/sos - Error code - 0x000021BE
13142정성태10/18/20225331도서: 시작하세요! C# 12 프로그래밍
13141정성태10/17/20226898.NET Framework: 2058. [in,out] 배열을 C#에서 C/C++로 넘기는 방법 - 세 번째 이야기파일 다운로드1
13140정성태10/11/20226269C/C++: 159. C/C++ - 리눅스 환경에서 u16string 문자열을 출력하는 방법 [2]
13139정성태10/9/20226075.NET Framework: 2057. 리눅스 환경의 .NET Core 3/5+ 메모리 덤프로부터 모든 닷넷 모듈을 추출하는 방법파일 다운로드1
13138정성태10/8/20227367.NET Framework: 2056. C# - await 비동기 호출을 기대한 메서드가 동기로 호출되었을 때의 부작용 [1]
13137정성태10/8/20225726.NET Framework: 2055. 리눅스 환경의 .NET Core 3/5+ 메모리 덤프로부터 닷넷 모듈을 추출하는 방법
13136정성태10/7/20226304.NET Framework: 2054. .NET Core/5+ SDK 설치 없이 dotnet-dump 사용하는 방법
13135정성태10/5/20226542.NET Framework: 2053. 리눅스 환경의 .NET Core 3/5+ 메모리 덤프를 분석하는 방법 - 두 번째 이야기
13134정성태10/4/20225273오류 유형: 820. There is a problem with AMD Radeon RX 5600 XT device. For more information, search for 'graphics device driver error code 31'
13133정성태10/4/20225596Windows: 211. Windows - (commit이 아닌) reserved 메모리 사용량 확인 방법 [1]
... 16  17  18  [19]  20  21  22  23  24  25  26  27  28  29  30  ...