LOADING

加载过慢请开启缓存 浏览器默认开启

C++ Template 阅读笔记(1)

2025/3/2 Language C++

最近在学习 C++ Template 这本书,补齐了一些模板元编程相关的信息,后续会持续更新。

函数模板

推断函数模板的返回值

假设我们希望计算两个两个类型不相同的变量中的最大值(如 1u 和 2.0f),我们要如何定义函数模板?

// since c++ 14
template <typename T1, typename T2> auto max(T1 x, T2 y) {
  return x > y ? x : y;
}

// here x's type is float
auto x = ::max(1u, 2.0f);

从 c++ 14 开始,我们可以使用 auto 让编译器帮我们自动计算返回值的类型,然而在此之前,我们必须使用 decltypecommon_type 这样的工具来告诉编译器,我们需要返回什么类型:

// c++ 11
template <typename T1, typename T2>
auto max(T1 x, T2 y) -> decltype(x > y ? x : y) {
  return x > y ? x : y;
}

// equal to
template <typename T1, typename T2>
auto max(T1 x, T2 y) -> decltype(true ? x : y) {
  return x > y ? x : y;
}

然而,上述的代码可能面临一些严重的问题。在一些特殊场景下,返回值可能为引用,这时候需要使用类型萃取(type trait)std::decay来将引用“退化”。

// c++ 11
#include <type_traits>
template <typename T1, typename T2>
auto max(T1 a, T2 b) -> typename std::decay<decltype(true ? a : b)>::type {
  return b < a ? a : b;
}

// since c++ 14, simplified
template <typename T1, typename T2>
auto max(T1 a, T2 b) -> std::decay_t<decltype(true ? a : b)> {
  return b < a ? a : b;
}

注意到 c++ 14 对 typename std::decay<decltype(true ? a : b)>::type 这样的写法进行了简化。

这边顺便回顾一下 autodecltype 的区别:

auto 在大部分场景下和模板类型 T 的行为是一致的,会对推导的类型进行退化(decay)。此处的退化是指:

  • 去除 constvolatile 等关键字;
  • 去除引用
  • 将数组退化成指针

decltype 则是老老实实地获取类型,不会进行退化操作。有一个比较有意思的特殊场景:

// will return int &
auto f(int a) -> decltype((a)) { return a; }

虽然上述的用法是不正确的(返回了垂悬引用,dangling reference),但是能表明 decltype 会把 () 括起来的变量视作引用。

言归正传,之前的写法用 decltype 虽然能解决问题,但是未免太麻烦了点,我们可以用 std::common_type_t<T1, T2, ...> 来解决这个问题:

// since c++ 14
template <typename T1, typename T2>
auto max(T1 a, T2 b) -> std::common_type_t<T1, T2> {
  return b < a ? a : b;
}

还有一种写法,实际上 c++ 在实例化模板时,只要能保证能推断出所有的类型即可,我们可以定义一个返回类型 RT:

template <typename RT, typename T1, typename T2> 
RT max(T1 a, T2 b) {
  return b < a ? a : b;
}
auto x = ::max<float>(1u, 2.0f);

这样只需要指定 RT 即可,因为 T1T2 的类型对于编译器而言是已知的。但是下面的写法就会出错:

template <typename T1, typename T2, typename RT>
RT max(T1 a, T2 b) {
  return b < a ? a : b;
}
auto x = ::max<unsigned int, float>(1u, 2.0f);

报错信息为:

Candidate template ignored: couldn’t infer template argument ‘RT’

编译器无法自动推断 RT 的类型,除非使用上述的 auto

c++ 还支持模板类型的默认值,比如常用的 std::priority_queue 实际上有 3 个模板类型。

类模板

c++ 支持使用基础的数值类型(double 除外)作为模板参数:

template <typename T, size_t MaxSize> class Stack {
  array<T, MaxSize> elements;
  Stack(T first) : elements({first}) {}
};

虽然不支持直接使用字符串常量,但是可以通过定义一个变量来解决

template <char const *name> class MyClass {
  ...
};
MyClass<"hello"> x; // ERROR: string literal "hello" not allowed
extern char const s03[] = "hi"; // external linkage 
char const s11[] = "hi"; // internal linkage
int main() {
  MyClass<s03> m03;               // OK (all versions)
  MyClass<s11> m11;               // OK since C++11
  static char const s17[] = "hi"; // no linkage
  MyClass<s17> m17;               // OK since C++17
}

推断指引(Deduction Guides)

template <typename T> class Stack {
  vector<T> elements;

public:
  Stack() = default;
  Stack(T first) : elements({first}) {}
};

Stack(char const *)->Stack<std::string>;

Stack s("123");

如果没有 Stack(char const *)->Stack<std::string>;,s 将被推导为 Stack<const char *>

Concept

template <typename T> struct AccumulationTraits;
template <> struct AccumulationTraits<int> {
  using AccT = int;
  static AccT constexpr zero = 0;
};

template <> struct AccumulationTraits<string> {
  using AccT = string;
  // support since c++ 17
  inline static AccT const zero = "";
};

template <typename T> auto accum(T begin, T end) {
  using AccT = AccumulationTraits<T>::AccT;
  AccT total = AccumulationTraits<T>::zero;

  for (T i = begin; i <= end; i++) {
    total += i;
  }
  return total;
}

int main(int, char **) {
  auto total = accum(1, 100);
  cout << total << endl;
}
template <typename T>
concept Addable = requires(T a, T b) {
                    { a + b } -> std::same_as<T>;
                  };

template <typename T>
  requires Addable<T> && std::is_arithmetic_v<T>
T accum(T begin, T end) {
  T total{};
  for (T i = begin; i <= end; i++) {
    total += i;
  }
  return total;
}

此处也可以把 typename 替换成 Addable 之类的 concept。和 rusttrait 语法非常类似。