IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> Tricky Basics of C++ Templates -> 正文阅读

[C++知识库]Tricky Basics of C++ Templates

这章将介绍关于模板的一些看起来有点棘手或者诡异的知识点,比如 typename 的其他用途、零初始化、函数模板的字符串字面值参数等等。这些知识点在你常年的模板实践中总会遇到。

关键字 typename

在模板内部,可以使用关键字 typename 指明一个标识符为一个类型。例如:

template<typename T>
class MyClass {
  public:
    ...
    void foo() {
      typename T::SubType* ptr;
    }
};

使用 typename 告诉编译器 SubType 是类内定义的一个类型。没有关键字 typename 前缀的话,编译器会将 SubType 当作一个没有类型的数据成员,而 T::SubType* ptr 会被认为是一个乘法表达式。

零初始化

对于作用域内的内置类型(int,double,pointer,etc.),没有显示进行初始化赋值时,将会产生一个未定义的值。

void foo()
{
  int x; // x has undefined value
  int* ptr; // ptr points to anywhere (instead of nowhere)
}

对于模板,如果模板类型是内置类型,也有同样的问题。

template<typename T>
void foo()
{
  T x; // x has undefined value if T is built-in type
}

为了解决这个问题,需要显示地为内置类型调用一个默认构造函数将其初始化为零(or false for bool,or nullptr for pointer)。

template<typename T>
void foo()
{
  T x{}; // x is zero (or false) if T is a built-in type
   // T x = T();  // before C++11
}

对于类的成员变量也是如此。

template<typename T>
class MyClass {
  private:
    T x;
  public:
    MyClass() : x{} { // ensures that x is initialized even for built-in types
    // MyClass() : x() { // before C++11
    }
    ...
};

C++11 开始,支持直接对非静态成员变量初始化。

template<typename T>
class MyClass {
  private:
    T x{}; // zero-initialize x unless otherwise specified
    ...
};

使用 this->

对于派生类模板,调用基类的函数时,不一定是真正使用基类的该函数。例如:

class Base {
  public:
    void bar();
};

template<typename T>
class Derived : Base<T> {
  public:
    void foo() {
      bar(); // calls external bar() or error
    }
};

一种解决方法是使用基类的作用域限定:

template<typename T>
class Derived : Base<T> {
  public:
    void foo() {
      Base<T>::bar();
    }
};

另一种方法是使用 this->

template<typename T>
class Derived : Base<T> {
  public:
    void foo() {
      this->bar();
    }
};

原始数组和字符串字面值的模板

当传递原始数组或者字符串字面值给模板时需要格外小心。

如果模板参数申明为引用,入参不会发生退化(decay)。比如你传递一个 “hello”,它的类型为 char const[6],但是 “hello!” 的类型是 char const[7],二者不是同一个类型。

#include <iostream>

template<typename T>
const T& max(const T& a, const T& b)
{
  return a < b ? b : a;
}

int main()
{
  ::max("hello", "world");   // OK
  ::max("hello", "world!");  // error
}

如果是值传递,则会发生类型退化。例如字符串字面值会退化成 char const*。但它的缺点是失去了数组长度的信息。

针对原始数组和字符串字面值,可以专门提供模板:

template<typename T, int N, int M>
bool less (T(&a)[N], T(&b)[M])
{
  for (int i = 0; i<N && i<M; ++i) {
    if (a[i]<b[i]) return true;
    if (b[i]<a[i]) return false;
  }
  return N < M;
}

int x[] = {1, 2, 3};
int y[] = {1, 2, 3, 4, 5};
std::cout << less(x, y);         // true:T = int, N = 3,M = 5
std::cout << less("ab", "abc");  // true:T = const char, N = 3,M = 4

如果只支持字符串字面值,可以将模板参数 T 替换为 const char

template<typename T, int N, int M>
bool less (const char(&a)[N], const char(&b)[M])
{
  for (int i = 0; i<N && i<M; ++i) {
    if (a[i]<b[i]) return true;
    if (b[i]<a[i]) return false;
  }
  return N < M;
}

对于边界未知的数组,可能需要重载和偏特化:

#include <iostream>
template<typename T>
struct MyClass; // primary template

template<typename T, std::size_t SZ>
struct MyClass<T[SZ]> // partial specialization for arrays of known bounds
{
  static void print() { std::cout << "print() for T[" << SZ << "]\n"; }
};

template<typename T, std::size_t SZ>
struct MyClass<T(&)[SZ]> // partial spec. for references to arrays of known bounds
{
  static void print() { std::cout << "print() for T(&)[" << SZ << "]\n"; }
};

template<typename T>
struct MyClass<T[]> // partial specialization for arrays of unknown bounds
{
  static void print() { std::cout << "print() for T[]\n"; }
};

template<typename T>
struct MyClass<T(&)[]> // partial spec. for references to arrays of unknown bounds
{
  static void print() { std::cout << "print() for T(&)[]\n"; }
};

template<typename T>
struct MyClass<T*> // partial specialization for pointers
{
  static void print() { std::cout << "print() for T*\n"; }
};

template<typename T1, typename T2, typename T3>
void foo(int a1[7], int a2[], // pointers by language rules
         int (&a3)[42],       // reference to array of known bound
         int (&x0)[],         // reference to array of unknown bound
         T1 x1,               // passing by value decays
         T2& x2, T3&& x3)     // passing by reference
{
  MyClass<decltype(a1)>::print(); // uses MyClass<T*>
  MyClass<decltype(a2)>::print(); // uses MyClass<T*>
  MyClass<decltype(a3)>::print(); // uses MyClass<T(&)[SZ]>
  MyClass<decltype(x0)>::print(); // uses MyClass<T(&)[]>
  MyClass<decltype(x1)>::print(); // uses MyClass<T*>
  MyClass<decltype(x2)>::print(); // uses MyClass<T(&)[]>
  MyClass<decltype(x3)>::print(); // uses MyClass<T(&)[]>
}

int main()
{
  int a[42];
  MyClass<decltype(a)>::print(); // uses MyClass<T[SZ]>
  extern int x[]; // forward declare array
  MyClass<decltype(x)>::print(); // uses MyClass<T[]>
  foo(a, a, a, x, x, x, x);
}

int x[] = {0, 8, 15}; // define forward-declared array

输出为:

print() for T[42]
print() for T[]
print() for T*
print() for T*
print() for T(&)[42]
print() for T(&)[]
print() for T*
print() for T(&)[]
print() for T(&)[]

成员模板

类的成员也可以是模板,包括嵌套类和成员函数。一般情况下,不能用不同类型的类互相赋值。

Stack<int> intStack1, intStack2; // stacks for ints
Stack<float> floatStack;         // stack for floats
...
intStack1 = intStack2;   // OK: stacks have same type
floatStack = intStack1;  // ERROR: stacks have different types

可以定义一个赋值运算符的模板来实现不同类型的赋值(通过合适的类型转换)。

template<typename T>
class Stack {
  private:
    std::deque<T> elems; // elements
  public:
    void push(T const&); // push element
    void pop(); // pop element
    T const& top() const; // return top element
    bool empty() const { // return whether the stack is empty
      return elems.empty();
    }
    // assign stack of elements of type T2
    template<typename T2>
    Stack& operator= (Stack<T2> const&);
};

template<typename T>
template<typename T2>
Stack<T>& Stack<T>::operator= (Stack<T2> const& op2)
{
  Stack<T2> tmp(op2); // create a copy of the assigned stack
  elems.clear();      // remove existing elements
  while (!tmp.empty()) { // copy all elements
    elems.push_front(tmp.top());
    tmp.pop();
  }
  return *this;
}

为了获取用来赋值的源对象所有成员的访问权限,可以把其他的 stack 实例声明为友元。

template<typename T>
class Stack {
  private:
    std::deque<T> elems; // elements
  public:
    void push(T const&); // push element
    void pop(); // pop element
    T const& top() const; // return top element
    bool empty() const { // return whether the stack is empty
      return elems.empty();
    }
    // assign stack of elements of type T2
    template<typename T2>
    Stack& operator= (Stack<T2> const&);
    // to get access to private members of Stack<T2> for any type T2:
    template<typename> friend class Stack;
};

这样,如下的赋值运算符实现就称为可能:

template<typename T>
template<typename T2>
Stack<T>& Stack<T>::operator= (Stack<T2> const& op2)
{
  elems.clear(); // remove existing elements
  elems.insert(elems.begin(), // insert at the beginning
               op2.elems.begin(), // all elements from op2
               op2.elems.end());
  return *this;
}

如下不同类型的类就可以相互赋值了:

Stack<int> intStack; // stack for ints
Stack<float> floatStack; // stack for floats
...
floatStack = intStack; // OK: stacks have different types,
                       // but int converts to float

当然,v.emplace_front(tmp.top()); 会在编译时候检查类型转换是否合适,否则会编译报错。例如:

Stack<std::string> stringStack; // stack of strings
Stack<float> floatStack; // stack of floats
...
floatStack = stringStack; // ERROR: std::string doesn’t convert to float

更进一步,你还可以改变内部的赋值的容器实现:

template<typename T, typename Cont = std::deque<T>>
class Stack {
  private:
    Cont elems; // elements
  public:
    void push(T const&); // push element
    void pop(); // pop element
    T const& top() const; // return top element
    bool empty() const { // return whether the stack is empty
      return elems.empty();
   }
   
  // assign stack of elements of type T2
  template<typename T2, typename Cont2>
  Stack& operator= (Stack<T2,Cont2> const&);
  // to get access to private members of Stack<T2> for any type T2:
  template<typename, typename> friend class Stack;
};

template<typename T, typename Cont>
template<typename T2, typename Cont2>
Stack<T,Cont>&
Stack<T,Cont>::operator= (Stack<T2,Cont2> const& op2)
{
  elems.clear(); // remove existing elements
  elems.insert(elems.begin(), // insert at the beginning
               op2.elems.begin(), // all elements from op2
               op2.elems.end());
  return *this;
}

成员函数模板的特化

成员函数模板也能偏特化或全特化。

#include<iostream>
class BoolString {
private:
std::string value;
  public:
    BoolString (std::string const& s)
      : value(s) {
    }
    
    template<typename T = std::string>
    T get() const { // get value (converted to T)
      return value;
    }
};

// full specialization for BoolString::getValue<>() for bool
template<>
inline bool BoolString::get<bool>() const {
  return value == "true" || value == "1" || value == "on";
}

int main() {
  std::cout << std::boolalpha;
  BoolString s1("hello");
  std::cout << s1.get() << '\n'; // prints hello
  std::cout << s1.get<bool>() << '\n'; // prints false
  BoolString s2("on");
  std::cout << s2.get<bool>() << '\n'; // prints true
  return 0;
}

.template 标识

调用一个成员模板时,显式限定模板实参有时候是有必要的,使用关键字 template 来确保 < 是模板实参列表的开始。例如:

template<unsigned long N>
void printBitset (std::bitset<N> const& bs) {
  std::cout << bs.template to_string<char, std::char_traits<char>,
                                     std::allocator<char>>();
}

.template 只需要用于依赖于模板参数的名称之后,这里的 b 依赖于模板参数 N

泛型lambda和成员模板

C++14 引入的泛型 lambda,其实就是成员模板的简化。例如:

[] (auto x, auto y) {
  return x + y;
}

// 等价如下
class SomeCompilerSpecificName {
  public:
    SomeCompilerSpecificName(); // constructor only callable by compiler
    template<typename T1, typename T2>
    auto operator() (T1 x, T2 y) const {
      return x + y;
    }
};

变量模板

C++14 开始支持变量模板,也即变量的类型也可以参数化。例如:

template<typename T>
constexpr T pi{3.1415926535897932385};

和所有模板一样,这个声明不应该出现在函数或局部作用域内。使用如下:

std::cout << pi<double> << '\n';
std::cout << pi<float> << '\n';

可以在不同编译单元中申明和使用变量模板。

// header.hpp:
template<typename T> T val{}; // zero initialized value

// translation unit 1:
#include "header.hpp"
int main()
{
  val<long> = 42;
  print();
}

// translation unit 2:
#include "header.hpp"
void print()
{
  std::cout << val<long> << '\n'; // OK: prints 42
}

变量模板也可以有默认类型。例如:

template<typename T = long double>
constexpr T pi = T{3.1415926535897932385};

std::cout << pi<> << '\n'; // outputs a long double
std::cout << pi<float> << '\n'; // outputs a float
std::cout << pi << '\n'; // ERROR

变量模板也支持非类型参数。

#include <iostream>
#include <array>
template<int N>
std::array<int,N> arr{}; // array with N elements, zero-initialized

template<auto N>
constexpr decltype(N) dval = N; // type of dval depends on passed value

int main()
{
  std::cout << dval<'c'> << '\n'; // N has value 'c' of type char
  arr<10>[0] = 42; // sets first element of global arr
  for (std::size_t i = 0; i < arr<10>.size(); ++i) { // uses values set in arr
    std::cout << arr<10>[i] << '\n';
  }
}

变量模板的一个用法是为类模板成员定义变量。

例如,有一个类:

template<typename T>
class MyClass {
  public:
    static constexpr int max = 1000;
};

template<typename T>
int myMax = MyClass<T>::max;

// 可以直接写
auto i = myMax<std::string>;
// 代替下面的写法
auto i = MyClass<std::string>::max;

类似的例子:

namespace std {
  template<typename T>
  class numeric_limits {
    public:
      ...
      static constexpr bool is_signed = false;
      ...
  };
}

template<typename T>
constexpr bool isSigned = std::numeric_limits<T>::is_signed;

// 可以直接写
isSigned<char>
// 代替下面的写法
std::numeric_limits<char>::is_signed

C++17 标准库就利用了变量模板简化了 type traits 的生成值

namespace std {
  template<typename T> constexpr bool is_const_v = is_const<T>::value;
}

std::is_const_v<T>       // since C++17
std::is_const<T>::value  // since C++11

模板的模板参数

模板参数自己也可以是一个类模板。例如前面的栈类模板的例子,用模板的模板参数,可以只指定容器类型而不需要指定元素类型。

Stack<int, std::vector<int>> vStack; // integer stack that uses a vector
Stack<int, std::vector> vStack; // integer stack that uses a vector

因此,需要将第二个模板参数指定为模板的模板参数:

template<typename T,
         template<typename Elem> class Cont = std::deque>
class Stack {
 public:
   void push(const T&);
   void pop();
   const T& top() const;
   bool empty() const { return elems.empty(); }
 private:
   Cont<T> elems;
};

C++17 开始可以使用 typename 修饰模板的模板参数中的 Cont

template<typename T,
         template<typename Elem> typename Cont = std::deque>
class Stack { // ERROR before C++17
  ...
};

由于模板的模板参数中的模板参数没有被使用(Cont 没有用到模板参数 Elem),因而可以简写为:

template<typename T,
         template<typename> class Cont = std::deque>
class Stack {
 ...
};

模板的模板实参匹配

容器还有另一个参数,即内存分配器。

template<typename T,
         template<typename Elem,
                  typename Alloc = std::allocator<Elem>>
         class Cont = std::deque>
class Stack {
  private:
    Cont<T> elems; // elements
    ...
};

Alloc 没被使用,也可以被省略。最终版本的 Stack 模板实现如下:

#include <deque>
#include <cassert>
#include <memory>
template<typename T,
         template<typename Elem,
                  typename = std::allocator<Elem>
                 >class Cont = std::deque>
class Stack {
  private:
    Cont<T> elems; // elements
    
  public:
    void push(T const&); // push element
    void pop(); // pop element
    T const& top() const; // return top element
    bool empty() const { // return whether the stack is empty
      return elems.empty();
    }
    
   // assign stack of elements of type T2
   template<typename T2,
            template<typename Elem2,
                     typename = std::allocator<Elem2>
                    >class Cont2>
  Stack<T,Cont>& operator= (Stack<T2,Cont2> const&);
  // to get access to private members of any Stack with elements of type T2:
  template<typename, template<typename, typename>class>
  friend class Stack;
};

template<typename T, template<typename,typename> class Cont>
void Stack<T,Cont>::push (T const& elem)
{
  elems.push_back(elem); // append copy of passed elem
}

template<typename T, template<typename,typename> class Cont>
void Stack<T,Cont>::pop ()
{
  assert(!elems.empty());
  elems.pop_back(); // remove last element
}

template<typename T, template<typename,typename> class Cont>
T const& Stack<T,Cont>::top () const
{
  assert(!elems.empty());
  return elems.back(); // return copy of last element
}

template<typename T, template<typename,typename> class Cont>
template<typename T2, template<typename,typename> class Cont2>
Stack<T,Cont>&
Stack<T,Cont>::operator= (Stack<T2,Cont2> const& op2)
{
  elems.clear(); // remove existing elements
  elems.insert(elems.begin(), // insert at the beginning
               op2.elems.begin(), // all elements from op2
               op2.elems.end());
  return *this;
}

int main()
{
  Stack<int> iStack; // stack of ints
  Stack<float> fStack; // stack of floats
  
  // manipulate int stack
  iStack.push(1);
  iStack.push(2);
  std::cout << "iStack.top(): " << iStack.top() << '\n';
  
  // manipulate float stack:
  fStack.push(3.3);
  std::cout << "fStack.top(): " << fStack.top() << '\n';
  
  // assign stack of different type and manipulate again
  fStack = iStack;
  fStack.push(4.4);
  std::cout << "fStack.top(): " << fStack.top() << '\n';

  // stack for doubless using a vector as an internal container
  Stack<double, std::vector> vStack;
  vStack.push(5.5);
  vStack.push(6.6);
  std::cout << "vStack.top(): " << vStack.top() << '\n';
  vStack = fStack;
  std::cout << "vStack: ";
  while (! vStack.empty()) {
    std::cout << vStack.top() << ' ';
    vStack.pop();
  }
  std::cout << '\n';
}

输出如下:

iStack.top(): 2
fStack.top(): 3.3
fStack.top(): 4.4
vStack.top(): 6.6
vStack: 4.4 2 1

引用

  • http://www.tmplbook.com
  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-07-05 23:22:06  更:2022-07-05 23:23:39 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年5日历 -2024/5/11 15:49:38-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码