溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

C++模板的特例化編譯為多重定義問題

發(fā)布時間:2020-08-02 21:37:38 來源:網(wǎng)絡(luò) 閱讀:2508 作者:zmh009_NAME 欄目:編程語言

    之前遇到了一些關(guān)于模板特例化的問題,自己總結(jié)一下。

    模板的特例化是C++新標(biāo)準的一個特點,可以自定義某些模板的實現(xiàn),比如在比較函數(shù)compare可以使用less<T>標(biāo)準庫模板比較string、int、char、指針等類型,但如果有const char*類型且比較字符串的字典大小時,就與之前的比較方式不同了:

#ifndef A_H
#define A_H

#include <iostream>
#include <cstring>
using std::less;
template <typename T>
int compare(const T v1,const T v2) 
{
    
    if (less<T>()(v1,v2)) 
    {
        return -1;
    }
    else if (less<T>()(v2,v1))
    {
        return 1;
    }
    else
    {
        return 0;
    }
}


template <>
inline int compare(const char *const a,const char *const b) 
{
    return strcmp(a,b);
}

#endif 


#include "a.h"
using std::cout;
using std::endl;

int main()
{
    cout << compare(1,2) << endl;  
    cout << compare("123","asd") << endl;  
    return 0;
}

    現(xiàn)在看上去沒什么問題,并且只有這兩個文件編譯通過且正常運行,如果多了一個包含a.h的文件a.cpp:

#include "a.h"

    這里只有一句,但包含了a.h文件表示將其模板函數(shù)的定義包含,且特例化的函數(shù)類似一個普通函數(shù),則a.cpp、main.cpp包含多個相同函數(shù)的定義,因此在鏈接時有重定義問題。

    解決方法:

    使用內(nèi)聯(lián)inline聲明特例化的模板,則某些函數(shù)的定義可以在多個文件包含(一些函數(shù)的實現(xiàn)可能不支持內(nèi)聯(lián)): 

template <>
inline int compare(const char *a,const char *b) 
{
    return strcmp(a,b);
}

    另外一個是使用一個文件包含該頭文件,讓所有鏈接的文件只有一個特例化定義

    第三個是定義普通同名函數(shù),通過重載調(diào)用非模板函數(shù)(在參數(shù)匹配級別相同時,非模板重載函數(shù)優(yōu)先調(diào)用)

// a.h
#ifndef A_H
#define A_H

#include <iostream>
#include <cstring>
using std::less;
template <typename T>
int compare(const T v1,const T v2)
{

    if (less<T>()(v1,v2))
    {
        return -1;
    }
    else if (less<T>()(v2,v1))
    {
        return 1;
    }
    else
    {
        return 0;
    }
}

int compare(const char *a,const char *b) ;

// a.cpp
#include "a.h"

int compare(const char *a, const char *b)
{
    std::cout << "const char*"<<std::endl;
    return strcmp(a,b);
}
// main.cpp 相同

    如果有其他的方法請大家一起交流

向AI問一下細節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI