如何避免 #includ 借助<sstream>模板功能?

How can I avoid #includ-ing <sstream> thanks to a template function?

本文关键字:功能 gt sstream 借助 何避免 lt #includ      更新时间:2023-10-16

我写了一个非常简单的日志记录类。

日志记录类有一个如下所示的函数:

template <typename T, typename... Args>
void log (T && first, Args &&... rest) {
  log (std::forward <T>    (first));
  log (std::forward <Args> (rest)...);
}
template <typename T>
void log (T && last) {
  /* use <sstream>'s std::ostringstream to convert each T to its string
     representation, then print it via a callback. */
}

由于我的日志记录类可能在整个程序中经常使用,因此我正在尝试减少由于#includ记录器文件而附加到每个翻译单元上的代码量。

我已经设法将除<string>之外的所有 #include 语句移出头文件并移入实现文件,除了整个 <sstream> 及其所有依赖项,这是实现此模板成员函数所必需的。

我已经考虑过显式实例化我需要的所有重载,但如果我希望将我的记录器与字符串文字一起使用,那么我想我必须为 const char (&) [n] 中的每个n显式实例化一个模板。 或者#include <type_traits> std::decay. :(

是否可以避免将<sstream>拖入使用我的记录器的所有 TU? 如何?

在我的项目中,我合并的一种方法是使用 C++11 中的std::to_string()。这可以避免使用<sstream>。您可能只需要包含<string>标头。对于特定的类,可以实现自己的公共方法to_string()

但是Android版本(我们一直使用到2016年年中(不幸的是,即使使用C++11也没有std::to_string()。在这种情况下,我们必须依靠 sstream .但是,您可以通过使用以下template和宏技巧(玩具代码(来避免在头文件中包含<sstream>

// header.h
template<typename ostringstream, typename T>
std::string my_to_string (T&& value)
{
  ostringstream oss;
  oss << value;
  return oss.str();
}
#define my_to_string(X) my_to_string<std::ostringstream>(X)

现在,在所有实现中,您必须按如下方式<sstream>和使用:

// implementation.cpp
#include<sstream>
int main ()
{
  double d = 1.2345;
  std::string s = my_to_string(d);
  std::cout << "converted: " << s << "n";
}

演示。