9

私はこの関数ヘッダーを持っています:

template <
    bool src_alpha,
    int sbpp, int dbpp,
    typename T1, typename T2,
    Color (*getFunc)(T1 data, Uint8* addr),
    void (*putFunc)(T2 data, Uint8* addr, Color c)
>
static void OperateOnSurfaces(T1 data1, T2 data2, SDL_Surface * bmpDest, SDL_Surface * bmpSrc, SDL_Rect& rDest, SDL_Rect& rSrc)

これが私がそれを使う方法です:

OperateOnSurfaces<
    true,
    32, 32,
    SDL_PixelFormat*, SDL_PixelFormat*,
    GetPixel<true,32>, PutPixel<true,true,32> >(
    bmpSrc->format, bmpDest->format,
    bmpDest, bmpSrc, rDest, rSrc);

これはGetPixelPutPixel

template<bool alpha, int bpp>
static Color GetPixel(SDL_PixelFormat* format, Uint8* addr) { /* .. */ }

template<bool alpha, bool alphablend, int bpp>
static void PutPixel(SDL_PixelFormat* format, Uint8* addr, Color col) { /* .. */ }

そして、私はこのエラーを受け取ります:

note: candidate template ignored: invalid explicitly-specified argument for template parameter 'getFunc' [3]

なんで?

4

1 に答える 1

7

これらの機能はすべて無料の機能だと思います。フリー関数を宣言すると、static内部リンクが得られます。C ++ 03の非型テンプレートパラメータには、外部リンケージ†</sup>が必要です。関数の前を削除するだけです。static

template <
    bool src_alpha,
    int sbpp, int dbpp,
    typename T1, typename T2,
    char (*getFunc)(T1 data, unsigned* addr),
    void (*putFunc)(T2 data, unsigned* addr, char c)
>
void OperateOnSurfaces(){}

template<bool alpha, int bpp>
char GetPixel(void* format, unsigned* addr);

template<bool alpha, bool alphablend, int bpp>
void PutPixel(void* format, unsigned* addr, char col);

int main(){
    OperateOnSurfaces<
        true,
        32, 32,
        void*, void*,
        GetPixel<true,32>, PutPixel<true,true,32> >();
}

この変更された例は、C ++98およびC++11モードのClang3.1およびGCc4.4.5で正常にコンパイルされ、警告はありません。そのままにstaticしておくと、Clangで得られたものと同様のエラーとメモが表示され、GCCは重要な情報を吐き出します(右にスクロールすると、「外部リンクがありません」)。

15:02:38 $ g++ t.cpp
t.cpp: In function ‘int main()’:
t.cpp:21: error: ‘GetPixel<true, 32>’ is not a valid template argument for type ‘char (*)(void*, unsigned int*)’ because function ‘char GetPixel(void*, unsigned int*) [with bool alpha = true, int bpp = 32]’ has not external linkage
t.cpp:21: error: ‘PutPixel<true, true, 32>’ is not a valid template argument for type ‘void (*)(void*, unsigned int*, char)’ because function ‘void PutPixel(void*, unsigned int*, char) [with bool alpha = true, bool alphablend = true, int bpp = 32]’ has not external linkage
t.cpp:21: error: no matching function for call to ‘OperateOnSurfaces()’

(C++03) §14.3.2 [temp.arg.nontype] p1

非型、非テンプレートのテンプレートパラメータのテンプレート引数は、次のいずれかである必要があります。

  • [...]

  • 外部リンケージを持つオブジェクトまたは関数のアドレス[...]

  • [...]

C ++ 11は文言を変更し、内部リンケージを備えた関数も許可することに注意してください。

(C++11) §14.3.2 [temp.arg.nontype] p1

非型、非テンプレートのテンプレートパラメータのテンプレート引数は、次のいずれかである必要があります。

  • [...]

  • 静的ストレージ期間と外部または内部リンケージを持つオブジェクト、または外部または内部リンケージを持つ関数のアドレスを指定する定数式(5.19)[ ... ]

  • [...]

Clangは現在、C ++ 11モードではこれに準拠していませんが、外部リンケージのある機能のみを許可しています。

于 2011-12-25T14:07:11.310 に答える