【问题标题】:Use of template <class T> in C++ when declaring classes and functions [closed]声明类和函数时在 C++ 中使用模板 <class T> [关闭]
【发布时间】:2017-06-21 07:25:56
【问题描述】:

我对 C++ 中的模板这个话题还很陌生。为什么在下面的玩具示例代码中,我们必须在类和每个函数的名称前加上 template &lt;class T&gt;(这意味着我们为什么需要它)?

是否可以修改代码不到处使用template &lt;class T&gt;

#include <iostream>
#include <vector>
#include <cstdlib>
#include <string>
#include <stdexcept>

using namespace std;

template <class T>
class Stack {   private:
    vector<T> elements;

  public:
    void push(T const &);
    void pop();
    T top();
    bool empty(); };

template <class T>
void Stack<T>::push(T const &elem) {
    elements.push_back(elem); }

template <class T>
void Stack<T>::pop() {
    if (elements.empty()) {
        throw out_of_range("Stack<>::pop(): empty stack");
    } else {
        elements.pop_back();
    }
}

template <class T>
T Stack<T>::top() {
    if (empty()) {
        throw out_of_range("Stack<>::top(): empty stack");
    }
    return elements.back();
}

template <class T>
bool Stack<T>::empty() {
    return elements.empty();
}


int main() {
    try {
        Stack<int> intStack;       // Stack of ints
        Stack<string> stringStack; // Stack of strings

        // Manipulate integer stack
        intStack.push(7);
        cout << intStack.top() << endl;

        // Manipulate string stack
        stringStack.push("hello");
        cout << stringStack.top() << std::endl;
        stringStack.pop();
        stringStack.pop();
    }
    catch (exception const &ex) {
        cerr << "Exception: " << ex.what() << endl;
        return -1;
    }
}

【问题讨论】:

  • 什么是“模板头”?
  • @cerebrou 是的,您可以通过将所有出现的T 替换为intdouble 等具体类型来做到这一点。但是你为什么要这样做呢?它使类变得不那么有用。
  • 对 C++ 中的模板主题不熟悉?开始here
  • 问:“为什么我们必须在类和每个函数的名称前加上template &lt;class T&gt;”?答:因为这是模板的语法。问:什么是模板,我们为什么需要它们,我们如何使用它们? A:这里太宽泛了。
  • 我相信这个问题一点儿也不过分。 OP 不熟悉内联方法实现(类内实现),他/她只是注意到定义方法的“通常”方式在模板的情况下有很多重复的语法。询问如何摆脱这种语法是非常合理的。

标签: c++ templates


【解决方案1】:

C++ 是一种statically typed 语言。

既然是这种情况,假设我想要一个函数来检查两个值的最小值或最大值。

我可以创建如下函数:

int max(int a, int b) { return (a > b) ? a : b; }

但这仅适用于 int 类型。如果我想要一个用于doubleuint64_tFooBarClass,我需要创建额外的函数:

double max(double a, double b) { return (a > b) ? a : b; }
uint64_t max(uint64_t a, uint64_t b) { return (a > b) ? a : b; }
FooBarClass max(FooBarClass a, FooBarClass b) { return (a > b) ? a : b; }

为什么在下面的玩具示例代码中,我们必须在类和每个函数的名称前加上 template ?

在 C++ 中,template 是一种引入称为 "generics" 的概念的方法。使用泛型,您不再需要为每种类型创建一个函数,类型将从模板函数签名中推导出来。

所以,我可以写如下:

template < typename T > // the typename or class keywords are synonymous when declaring a template
T max(T a, T b) { return (a > b) ? a : b; }

在函数签名中包含template &lt; typename T &gt; 向编译器表明该函数是“通用”函数,必须进行相应处理。

另一方面,如果您只是满足以下条件:

T max(T a, T b) { return (a > b) ? a : b; }

编译器会期望T 是一个显式类型(如typedef int T;)。

模板可以同时应用于类或函数:

template < typename T >
class Wrapper {
    public:
        Wrapper(const T& val) : _val(val) {}
        // ... other code

        template < typename X >
        operator X()
        {
            return static_cast<X>(this->_val);
        }

        T operator+(const T& val)
        {
            return this->_val + val;
        }

        friend std::ostream& operator<<(std::ostream& os, const Wrapper& val)
        {
            os << val._val;
            return os;
        }

    private:
        T _val;
};

int main(int argc, char* argv[])
{
    Wrapper<int> x(42);
    // template deduction will call Wrapper::operator X() with X being a double
    double d = x;
    // template deduction will call Wrapper::operator X() with X being a long
    long l = x;
    // this actually calls T operator+(const T& val) with val = 10 and T = int
    int i = x + 10;

    std::cout << "x = " << x << std::endl  // 42
              << "d = " << d << std::endl  // 42
              << "l = " << l << std::endl  // 42
              << "i = " << i << std::endl; // 52
    return 0;
}

你也可以显式定义一个模板类型,这被称为template specialization 鉴于您目前对模板的理解,超出了此答案的范围。

是否可以修改代码不到处使用模板?

是和不是。是的,您可以删除 template &lt; class T &gt; 签名定义,但如前所述,类/函数的签名具有完全不同的含义。

希望这能澄清一点。如果使用得当,模板是 C++ 中非常强大的工具。

【讨论】:

  • C++ 是静态类型,而不是“强类型”。它允许隐式转换,这可能是有损的。
【解决方案2】:

您必须在每个成员定义中包含template &lt;class T&gt;,因为这是成员名称的一部分。

您可以在类模板主体中定义函数,只需一行template &lt;class T&gt;

template <class T> 
class Stack {   
  private:
    vector<T> elements;

  public:
    void push(T const &) {
      elements.push_back(elem); }
    void pop() {
      if (elements.empty()) {
          throw out_of_range("Stack<>::pop(): empty stack");
      } else {
          elements.pop_back();
      } }
    T top() {
      if (empty()) {
          throw out_of_range("Stack<>::top(): empty stack");
      }
      return elements.back(); }
    bool empty() {
      return elements.empty(); } };

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2019-07-30
    • 2018-05-02
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多