为什么用这种方式用C ++定义模板函数?

时间:2017-06-07 02:14:50

标签: c++ header

#include <openpose/utilities/errorAndLog.hpp>
#include <openpose/utilities/string.hpp>

namespace op
{
    template<typename T>
    std::string toFixedLengthString(const T number, const unsigned long long stringLength)
    {
        try
        {
            const auto numberAsString = std::to_string(number);
            if (stringLength > 0)
            {
                if (number < 0)
                    error("toFixedLengthString: number cannot be <= 0, in this case it is: " + numberAsString + ".", __LINE__, __FUNCTION__, __FILE__);

                const auto zerosToAdd = stringLength - numberAsString.size();
                if (zerosToAdd < 0)
                {
                    const auto errorMessage = "toFixedLengthString: number greater than maximum number of digits (stringLength): "
                                            + numberAsString + " vs. " + std::to_string(stringLength) + ".";
                    error(errorMessage, __LINE__, __FUNCTION__, __FILE__);
                }

                return { std::string(zerosToAdd, '0') + numberAsString};
            }
            else
                return numberAsString;
        }
        catch (const std::exception& e)
        {
            error(e.what(), __LINE__, __FUNCTION__, __FILE__);
            return "";
        }
    }

    // Signed
    template std::string toFixedLengthString<char>(const char number, const unsigned long long stringLength);
    template std::string toFixedLengthString<signed char>(const signed char number, const unsigned long long stringLength);
    template std::string toFixedLengthString<short>(const short number, const unsigned long long stringLength);
    template std::string toFixedLengthString<int>(const int number, const unsigned long long stringLength);
    template std::string toFixedLengthString<long>(const long number, const unsigned long long stringLength);
    template std::string toFixedLengthString<long long>(const long long number, const unsigned long long stringLength);
    // Unsigned
    template std::string toFixedLengthString<unsigned char>(const unsigned char number, const unsigned long long stringLength);
    template std::string toFixedLengthString<unsigned short>(const unsigned short number, const unsigned long long stringLength);
    template std::string toFixedLengthString<unsigned int>(const unsigned int number, const unsigned long long stringLength);
    template std::string toFixedLengthString<unsigned long>(const unsigned long number, const unsigned long long stringLength);
    template std::string toFixedLengthString<unsigned long long>(const unsigned long long number, const unsigned long long stringLength);
}

这是src文件,他在头文件中有相应的功能。他为什么需要定义下面的内容,而不是只将整个模板函数放在头文件中,根据SO上的其他帖子,应该是人们定义模板函数的方式。

2 个答案:

答案 0 :(得分:0)

(我是OpenPose的作者之一)

主要原因:如果我在hpp中定义它,那么每次对代码进行一些小改动时,都需要等待很多时间来编译代码的所有模板。

通过这种方式,这些模板只编译一次(第一次编译OpenPose时),而不是每次在代码中进行更改时都会编译。

附注:我接受建议和评论,当我不像C ++那样专业时,我自己为OpenPose提供了太多代码。

答案 1 :(得分:0)

如果未采用这种方式定义,则链接器将引发错误。另一种选择是将所有函数定义添加到头文件中。

请参阅:https://isocpp.org/wiki/faq/templates#separate-template-fn-defn-from-decl